3465 lines
106 KiB
JavaScript
3465 lines
106 KiB
JavaScript
import {
|
||
Link
|
||
} from "./chunk-MX7X7RGK.js";
|
||
import {
|
||
Decoration,
|
||
DecorationSet,
|
||
Extension,
|
||
Fragment,
|
||
Mapping,
|
||
Mark,
|
||
Node3,
|
||
NodeSelection,
|
||
Plugin,
|
||
PluginKey,
|
||
Selection,
|
||
Slice,
|
||
TextSelection,
|
||
callOrReturn,
|
||
canInsertNode,
|
||
dropPoint,
|
||
getExtensionField,
|
||
getNodeAtPosition,
|
||
getNodeType,
|
||
getRenderedAttributes,
|
||
isAtEndOfNode,
|
||
isAtStartOfNode,
|
||
isNodeActive,
|
||
isNodeEmpty,
|
||
isNodeSelection,
|
||
keydownHandler,
|
||
markInputRule,
|
||
markPasteRule,
|
||
mergeAttributes,
|
||
nodeInputRule,
|
||
parseIndentedBlocks,
|
||
renderNestedMarkdownContent,
|
||
textblockTypeInputRule,
|
||
wrappingInputRule
|
||
} from "./chunk-YRIELJS7.js";
|
||
import "./chunk-4MBMRILA.js";
|
||
|
||
// node_modules/@tiptap/core/dist/jsx-runtime/jsx-runtime.js
|
||
var h = (tag, attributes) => {
|
||
if (tag === "slot") {
|
||
return 0;
|
||
}
|
||
if (tag instanceof Function) {
|
||
return tag(attributes);
|
||
}
|
||
const { children, ...rest } = attributes != null ? attributes : {};
|
||
if (tag === "svg") {
|
||
throw new Error("SVG elements are not supported in the JSX syntax, use the array syntax instead");
|
||
}
|
||
return [tag, rest, children];
|
||
};
|
||
|
||
// node_modules/@tiptap/extension-blockquote/dist/index.js
|
||
var inputRegex = /^\s*>\s$/;
|
||
var Blockquote = Node3.create({
|
||
name: "blockquote",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
content: "block+",
|
||
group: "block",
|
||
defining: true,
|
||
parseHTML() {
|
||
return [{ tag: "blockquote" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return h("blockquote", { ...mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), children: h("slot", {}) });
|
||
},
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.createNode("blockquote", void 0, helpers.parseChildren(token.tokens || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
const prefix = ">";
|
||
const result = [];
|
||
node.content.forEach((child) => {
|
||
const childContent = h2.renderChildren([child]);
|
||
const lines = childContent.split("\n");
|
||
const linesWithPrefix = lines.map((line) => {
|
||
if (line.trim() === "") {
|
||
return prefix;
|
||
}
|
||
return `${prefix} ${line}`;
|
||
});
|
||
result.push(linesWithPrefix.join("\n"));
|
||
});
|
||
return result.join(`
|
||
${prefix}
|
||
`);
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setBlockquote: () => ({ commands }) => {
|
||
return commands.wrapIn(this.name);
|
||
},
|
||
toggleBlockquote: () => ({ commands }) => {
|
||
return commands.toggleWrap(this.name);
|
||
},
|
||
unsetBlockquote: () => ({ commands }) => {
|
||
return commands.lift(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Shift-b": () => this.editor.commands.toggleBlockquote()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
wrappingInputRule({
|
||
find: inputRegex,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-bold/dist/index.js
|
||
var starInputRegex = /(?:^|\s)(\*\*(?!\s+\*\*)((?:[^*]+))\*\*(?!\s+\*\*))$/;
|
||
var starPasteRegex = /(?:^|\s)(\*\*(?!\s+\*\*)((?:[^*]+))\*\*(?!\s+\*\*))/g;
|
||
var underscoreInputRegex = /(?:^|\s)(__(?!\s+__)((?:[^_]+))__(?!\s+__))$/;
|
||
var underscorePasteRegex = /(?:^|\s)(__(?!\s+__)((?:[^_]+))__(?!\s+__))/g;
|
||
var Bold = Mark.create({
|
||
name: "bold",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "strong"
|
||
},
|
||
{
|
||
tag: "b",
|
||
getAttrs: (node) => node.style.fontWeight !== "normal" && null
|
||
},
|
||
{
|
||
style: "font-weight=400",
|
||
clearMark: (mark) => mark.type.name === this.name
|
||
},
|
||
{
|
||
style: "font-weight",
|
||
getAttrs: (value) => /^(bold(er)?|[5-9]\d{2,})$/.test(value) && null
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return h("strong", { ...mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), children: h("slot", {}) });
|
||
},
|
||
markdownTokenName: "strong",
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.applyMark("bold", helpers.parseInline(token.tokens || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
return `**${h2.renderChildren(node)}**`;
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setBold: () => ({ commands }) => {
|
||
return commands.setMark(this.name);
|
||
},
|
||
toggleBold: () => ({ commands }) => {
|
||
return commands.toggleMark(this.name);
|
||
},
|
||
unsetBold: () => ({ commands }) => {
|
||
return commands.unsetMark(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-b": () => this.editor.commands.toggleBold(),
|
||
"Mod-B": () => this.editor.commands.toggleBold()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
markInputRule({
|
||
find: starInputRegex,
|
||
type: this.type
|
||
}),
|
||
markInputRule({
|
||
find: underscoreInputRegex,
|
||
type: this.type
|
||
})
|
||
];
|
||
},
|
||
addPasteRules() {
|
||
return [
|
||
markPasteRule({
|
||
find: starPasteRegex,
|
||
type: this.type
|
||
}),
|
||
markPasteRule({
|
||
find: underscorePasteRegex,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-code/dist/index.js
|
||
var inputRegex2 = /(^|[^`])`([^`]+)`(?!`)$/;
|
||
var pasteRegex = /(^|[^`])`([^`]+)`(?!`)/g;
|
||
var Code = Mark.create({
|
||
name: "code",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
excludes: "_",
|
||
code: true,
|
||
exitable: true,
|
||
parseHTML() {
|
||
return [{ tag: "code" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["code", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
markdownTokenName: "codespan",
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.applyMark("code", [{ type: "text", text: token.text || "" }]);
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return `\`${h2.renderChildren(node.content)}\``;
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setCode: () => ({ commands }) => {
|
||
return commands.setMark(this.name);
|
||
},
|
||
toggleCode: () => ({ commands }) => {
|
||
return commands.toggleMark(this.name);
|
||
},
|
||
unsetCode: () => ({ commands }) => {
|
||
return commands.unsetMark(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-e": () => this.editor.commands.toggleCode()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
markInputRule({
|
||
find: inputRegex2,
|
||
type: this.type
|
||
})
|
||
];
|
||
},
|
||
addPasteRules() {
|
||
return [
|
||
markPasteRule({
|
||
find: pasteRegex,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-code-block/dist/index.js
|
||
var DEFAULT_TAB_SIZE = 4;
|
||
var backtickInputRegex = /^```([a-z]+)?[\s\n]$/;
|
||
var tildeInputRegex = /^~~~([a-z]+)?[\s\n]$/;
|
||
var CodeBlock = Node3.create({
|
||
name: "codeBlock",
|
||
addOptions() {
|
||
return {
|
||
languageClassPrefix: "language-",
|
||
exitOnTripleEnter: true,
|
||
exitOnArrowDown: true,
|
||
defaultLanguage: null,
|
||
enableTabIndentation: false,
|
||
tabSize: DEFAULT_TAB_SIZE,
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
content: "text*",
|
||
marks: "",
|
||
group: "block",
|
||
code: true,
|
||
defining: true,
|
||
addAttributes() {
|
||
return {
|
||
language: {
|
||
default: this.options.defaultLanguage,
|
||
parseHTML: (element) => {
|
||
var _a;
|
||
const { languageClassPrefix } = this.options;
|
||
if (!languageClassPrefix) {
|
||
return null;
|
||
}
|
||
const classNames = [...((_a = element.firstElementChild) == null ? void 0 : _a.classList) || []];
|
||
const languages = classNames.filter((className) => className.startsWith(languageClassPrefix)).map((className) => className.replace(languageClassPrefix, ""));
|
||
const language = languages[0];
|
||
if (!language) {
|
||
return null;
|
||
}
|
||
return language;
|
||
},
|
||
rendered: false
|
||
}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "pre",
|
||
preserveWhitespace: "full"
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ node, HTMLAttributes }) {
|
||
return [
|
||
"pre",
|
||
mergeAttributes(this.options.HTMLAttributes, HTMLAttributes),
|
||
[
|
||
"code",
|
||
{
|
||
class: node.attrs.language ? this.options.languageClassPrefix + node.attrs.language : null
|
||
},
|
||
0
|
||
]
|
||
];
|
||
},
|
||
markdownTokenName: "code",
|
||
parseMarkdown: (token, helpers) => {
|
||
var _a;
|
||
if (((_a = token.raw) == null ? void 0 : _a.startsWith("```")) === false && token.codeBlockStyle !== "indented") {
|
||
return [];
|
||
}
|
||
return helpers.createNode(
|
||
"codeBlock",
|
||
{ language: token.lang || null },
|
||
token.text ? [helpers.createTextNode(token.text)] : []
|
||
);
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
var _a;
|
||
let output = "";
|
||
const language = ((_a = node.attrs) == null ? void 0 : _a.language) || "";
|
||
if (!node.content) {
|
||
output = `\`\`\`${language}
|
||
|
||
\`\`\``;
|
||
} else {
|
||
const lines = [`\`\`\`${language}`, h2.renderChildren(node.content), "```"];
|
||
output = lines.join("\n");
|
||
}
|
||
return output;
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setCodeBlock: (attributes) => ({ commands }) => {
|
||
return commands.setNode(this.name, attributes);
|
||
},
|
||
toggleCodeBlock: (attributes) => ({ commands }) => {
|
||
return commands.toggleNode(this.name, "paragraph", attributes);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Alt-c": () => this.editor.commands.toggleCodeBlock(),
|
||
// remove code block when at start of document or code block is empty
|
||
Backspace: () => {
|
||
const { empty, $anchor } = this.editor.state.selection;
|
||
const isAtStart = $anchor.pos === 1;
|
||
if (!empty || $anchor.parent.type.name !== this.name) {
|
||
return false;
|
||
}
|
||
if (isAtStart || !$anchor.parent.textContent.length) {
|
||
return this.editor.commands.clearNodes();
|
||
}
|
||
return false;
|
||
},
|
||
// handle tab indentation
|
||
Tab: ({ editor }) => {
|
||
var _a;
|
||
if (!this.options.enableTabIndentation) {
|
||
return false;
|
||
}
|
||
const tabSize = (_a = this.options.tabSize) != null ? _a : DEFAULT_TAB_SIZE;
|
||
const { state } = editor;
|
||
const { selection } = state;
|
||
const { $from, empty } = selection;
|
||
if ($from.parent.type !== this.type) {
|
||
return false;
|
||
}
|
||
const indent = " ".repeat(tabSize);
|
||
if (empty) {
|
||
return editor.commands.insertContent(indent);
|
||
}
|
||
return editor.commands.command(({ tr }) => {
|
||
const { from: from2, to } = selection;
|
||
const text = state.doc.textBetween(from2, to, "\n", "\n");
|
||
const lines = text.split("\n");
|
||
const indentedText = lines.map((line) => indent + line).join("\n");
|
||
tr.replaceWith(from2, to, state.schema.text(indentedText));
|
||
return true;
|
||
});
|
||
},
|
||
// handle shift+tab reverse indentation
|
||
"Shift-Tab": ({ editor }) => {
|
||
var _a;
|
||
if (!this.options.enableTabIndentation) {
|
||
return false;
|
||
}
|
||
const tabSize = (_a = this.options.tabSize) != null ? _a : DEFAULT_TAB_SIZE;
|
||
const { state } = editor;
|
||
const { selection } = state;
|
||
const { $from, empty } = selection;
|
||
if ($from.parent.type !== this.type) {
|
||
return false;
|
||
}
|
||
if (empty) {
|
||
return editor.commands.command(({ tr }) => {
|
||
var _a2;
|
||
const { pos } = $from;
|
||
const codeBlockStart = $from.start();
|
||
const codeBlockEnd = $from.end();
|
||
const allText = state.doc.textBetween(codeBlockStart, codeBlockEnd, "\n", "\n");
|
||
const lines = allText.split("\n");
|
||
let currentLineIndex = 0;
|
||
let charCount = 0;
|
||
const relativeCursorPos = pos - codeBlockStart;
|
||
for (let i = 0; i < lines.length; i += 1) {
|
||
if (charCount + lines[i].length >= relativeCursorPos) {
|
||
currentLineIndex = i;
|
||
break;
|
||
}
|
||
charCount += lines[i].length + 1;
|
||
}
|
||
const currentLine = lines[currentLineIndex];
|
||
const leadingSpaces = ((_a2 = currentLine.match(/^ */)) == null ? void 0 : _a2[0]) || "";
|
||
const spacesToRemove = Math.min(leadingSpaces.length, tabSize);
|
||
if (spacesToRemove === 0) {
|
||
return true;
|
||
}
|
||
let lineStartPos = codeBlockStart;
|
||
for (let i = 0; i < currentLineIndex; i += 1) {
|
||
lineStartPos += lines[i].length + 1;
|
||
}
|
||
tr.delete(lineStartPos, lineStartPos + spacesToRemove);
|
||
const cursorPosInLine = pos - lineStartPos;
|
||
if (cursorPosInLine <= spacesToRemove) {
|
||
tr.setSelection(TextSelection.create(tr.doc, lineStartPos));
|
||
}
|
||
return true;
|
||
});
|
||
}
|
||
return editor.commands.command(({ tr }) => {
|
||
const { from: from2, to } = selection;
|
||
const text = state.doc.textBetween(from2, to, "\n", "\n");
|
||
const lines = text.split("\n");
|
||
const reverseIndentText = lines.map((line) => {
|
||
var _a2;
|
||
const leadingSpaces = ((_a2 = line.match(/^ */)) == null ? void 0 : _a2[0]) || "";
|
||
const spacesToRemove = Math.min(leadingSpaces.length, tabSize);
|
||
return line.slice(spacesToRemove);
|
||
}).join("\n");
|
||
tr.replaceWith(from2, to, state.schema.text(reverseIndentText));
|
||
return true;
|
||
});
|
||
},
|
||
// exit node on triple enter
|
||
Enter: ({ editor }) => {
|
||
if (!this.options.exitOnTripleEnter) {
|
||
return false;
|
||
}
|
||
const { state } = editor;
|
||
const { selection } = state;
|
||
const { $from, empty } = selection;
|
||
if (!empty || $from.parent.type !== this.type) {
|
||
return false;
|
||
}
|
||
const isAtEnd = $from.parentOffset === $from.parent.nodeSize - 2;
|
||
const endsWithDoubleNewline = $from.parent.textContent.endsWith("\n\n");
|
||
if (!isAtEnd || !endsWithDoubleNewline) {
|
||
return false;
|
||
}
|
||
return editor.chain().command(({ tr }) => {
|
||
tr.delete($from.pos - 2, $from.pos);
|
||
return true;
|
||
}).exitCode().run();
|
||
},
|
||
// exit node on arrow down
|
||
ArrowDown: ({ editor }) => {
|
||
if (!this.options.exitOnArrowDown) {
|
||
return false;
|
||
}
|
||
const { state } = editor;
|
||
const { selection, doc } = state;
|
||
const { $from, empty } = selection;
|
||
if (!empty || $from.parent.type !== this.type) {
|
||
return false;
|
||
}
|
||
const isAtEnd = $from.parentOffset === $from.parent.nodeSize - 2;
|
||
if (!isAtEnd) {
|
||
return false;
|
||
}
|
||
const after = $from.after();
|
||
if (after === void 0) {
|
||
return false;
|
||
}
|
||
const nodeAfter = doc.nodeAt(after);
|
||
if (nodeAfter) {
|
||
return editor.commands.command(({ tr }) => {
|
||
tr.setSelection(Selection.near(doc.resolve(after)));
|
||
return true;
|
||
});
|
||
}
|
||
return editor.commands.exitCode();
|
||
}
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
textblockTypeInputRule({
|
||
find: backtickInputRegex,
|
||
type: this.type,
|
||
getAttributes: (match) => ({
|
||
language: match[1]
|
||
})
|
||
}),
|
||
textblockTypeInputRule({
|
||
find: tildeInputRegex,
|
||
type: this.type,
|
||
getAttributes: (match) => ({
|
||
language: match[1]
|
||
})
|
||
})
|
||
];
|
||
},
|
||
addProseMirrorPlugins() {
|
||
return [
|
||
// this plugin creates a code block for pasted content from VS Code
|
||
// we can also detect the copied code language
|
||
new Plugin({
|
||
key: new PluginKey("codeBlockVSCodeHandler"),
|
||
props: {
|
||
handlePaste: (view, event) => {
|
||
if (!event.clipboardData) {
|
||
return false;
|
||
}
|
||
if (this.editor.isActive(this.type.name)) {
|
||
return false;
|
||
}
|
||
const text = event.clipboardData.getData("text/plain");
|
||
const vscode = event.clipboardData.getData("vscode-editor-data");
|
||
const vscodeData = vscode ? JSON.parse(vscode) : void 0;
|
||
const language = vscodeData == null ? void 0 : vscodeData.mode;
|
||
if (!text || !language) {
|
||
return false;
|
||
}
|
||
const { tr, schema } = view.state;
|
||
const textNode = schema.text(text.replace(/\r\n?/g, "\n"));
|
||
tr.replaceSelectionWith(this.type.create({ language }, textNode));
|
||
if (tr.selection.$from.parent.type !== this.type) {
|
||
tr.setSelection(TextSelection.near(tr.doc.resolve(Math.max(0, tr.selection.from - 2))));
|
||
}
|
||
tr.setMeta("paste", true);
|
||
view.dispatch(tr);
|
||
return true;
|
||
}
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-document/dist/index.js
|
||
var Document = Node3.create({
|
||
name: "doc",
|
||
topNode: true,
|
||
content: "block+",
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return h2.renderChildren(node.content, "\n\n");
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-hard-break/dist/index.js
|
||
var HardBreak = Node3.create({
|
||
name: "hardBreak",
|
||
markdownTokenName: "br",
|
||
addOptions() {
|
||
return {
|
||
keepMarks: true,
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
inline: true,
|
||
group: "inline",
|
||
selectable: false,
|
||
linebreakReplacement: true,
|
||
parseHTML() {
|
||
return [{ tag: "br" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["br", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes)];
|
||
},
|
||
renderText() {
|
||
return "\n";
|
||
},
|
||
renderMarkdown: () => `
|
||
`,
|
||
parseMarkdown: () => {
|
||
return {
|
||
type: "hardBreak"
|
||
};
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setHardBreak: () => ({ commands, chain, state, editor }) => {
|
||
return commands.first([
|
||
() => commands.exitCode(),
|
||
() => commands.command(() => {
|
||
const { selection, storedMarks } = state;
|
||
if (selection.$from.parent.type.spec.isolating) {
|
||
return false;
|
||
}
|
||
const { keepMarks } = this.options;
|
||
const { splittableMarks } = editor.extensionManager;
|
||
const marks = storedMarks || selection.$to.parentOffset && selection.$from.marks();
|
||
return chain().insertContent({ type: this.name }).command(({ tr, dispatch }) => {
|
||
if (dispatch && marks && keepMarks) {
|
||
const filteredMarks = marks.filter((mark) => splittableMarks.includes(mark.type.name));
|
||
tr.ensureMarks(filteredMarks);
|
||
}
|
||
return true;
|
||
}).run();
|
||
})
|
||
]);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Enter": () => this.editor.commands.setHardBreak(),
|
||
"Shift-Enter": () => this.editor.commands.setHardBreak()
|
||
};
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-heading/dist/index.js
|
||
var Heading = Node3.create({
|
||
name: "heading",
|
||
addOptions() {
|
||
return {
|
||
levels: [1, 2, 3, 4, 5, 6],
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
content: "inline*",
|
||
group: "block",
|
||
defining: true,
|
||
addAttributes() {
|
||
return {
|
||
level: {
|
||
default: 1,
|
||
rendered: false
|
||
}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return this.options.levels.map((level) => ({
|
||
tag: `h${level}`,
|
||
attrs: { level }
|
||
}));
|
||
},
|
||
renderHTML({ node, HTMLAttributes }) {
|
||
const hasLevel = this.options.levels.includes(node.attrs.level);
|
||
const level = hasLevel ? node.attrs.level : this.options.levels[0];
|
||
return [`h${level}`, mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.createNode("heading", { level: token.depth || 1 }, helpers.parseInline(token.tokens || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
var _a;
|
||
const level = ((_a = node.attrs) == null ? void 0 : _a.level) ? parseInt(node.attrs.level, 10) : 1;
|
||
const headingChars = "#".repeat(level);
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return `${headingChars} ${h2.renderChildren(node.content)}`;
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setHeading: (attributes) => ({ commands }) => {
|
||
if (!this.options.levels.includes(attributes.level)) {
|
||
return false;
|
||
}
|
||
return commands.setNode(this.name, attributes);
|
||
},
|
||
toggleHeading: (attributes) => ({ commands }) => {
|
||
if (!this.options.levels.includes(attributes.level)) {
|
||
return false;
|
||
}
|
||
return commands.toggleNode(this.name, "paragraph", attributes);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return this.options.levels.reduce(
|
||
(items, level) => ({
|
||
...items,
|
||
...{
|
||
[`Mod-Alt-${level}`]: () => this.editor.commands.toggleHeading({ level })
|
||
}
|
||
}),
|
||
{}
|
||
);
|
||
},
|
||
addInputRules() {
|
||
return this.options.levels.map((level) => {
|
||
return textblockTypeInputRule({
|
||
find: new RegExp(`^(#{${Math.min(...this.options.levels)},${level}})\\s$`),
|
||
type: this.type,
|
||
getAttributes: {
|
||
level
|
||
}
|
||
});
|
||
});
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-horizontal-rule/dist/index.js
|
||
var HorizontalRule = Node3.create({
|
||
name: "horizontalRule",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {},
|
||
nextNodeType: "paragraph"
|
||
};
|
||
},
|
||
group: "block",
|
||
parseHTML() {
|
||
return [{ tag: "hr" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["hr", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes)];
|
||
},
|
||
markdownTokenName: "hr",
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.createNode("horizontalRule");
|
||
},
|
||
renderMarkdown: () => {
|
||
return "---";
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setHorizontalRule: () => ({ chain, state }) => {
|
||
if (!canInsertNode(state, state.schema.nodes[this.name])) {
|
||
return false;
|
||
}
|
||
const { selection } = state;
|
||
const { $to: $originTo } = selection;
|
||
const currentChain = chain();
|
||
if (isNodeSelection(selection)) {
|
||
currentChain.insertContentAt($originTo.pos, {
|
||
type: this.name
|
||
});
|
||
} else {
|
||
currentChain.insertContent({ type: this.name });
|
||
}
|
||
return currentChain.command(({ state: chainState, tr, dispatch }) => {
|
||
if (dispatch) {
|
||
const { $to } = tr.selection;
|
||
const posAfter = $to.end();
|
||
if ($to.nodeAfter) {
|
||
if ($to.nodeAfter.isTextblock) {
|
||
tr.setSelection(TextSelection.create(tr.doc, $to.pos + 1));
|
||
} else if ($to.nodeAfter.isBlock) {
|
||
tr.setSelection(NodeSelection.create(tr.doc, $to.pos));
|
||
} else {
|
||
tr.setSelection(TextSelection.create(tr.doc, $to.pos));
|
||
}
|
||
} else {
|
||
const nodeType = chainState.schema.nodes[this.options.nextNodeType] || $to.parent.type.contentMatch.defaultType;
|
||
const node = nodeType == null ? void 0 : nodeType.create();
|
||
if (node) {
|
||
tr.insert(posAfter, node);
|
||
tr.setSelection(TextSelection.create(tr.doc, posAfter + 1));
|
||
}
|
||
}
|
||
tr.scrollIntoView();
|
||
}
|
||
return true;
|
||
}).run();
|
||
}
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
nodeInputRule({
|
||
find: /^(?:---|—-|___\s|\*\*\*\s)$/,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-italic/dist/index.js
|
||
var starInputRegex2 = /(?:^|\s)(\*(?!\s+\*)((?:[^*]+))\*(?!\s+\*))$/;
|
||
var starPasteRegex2 = /(?:^|\s)(\*(?!\s+\*)((?:[^*]+))\*(?!\s+\*))/g;
|
||
var underscoreInputRegex2 = /(?:^|\s)(_(?!\s+_)((?:[^_]+))_(?!\s+_))$/;
|
||
var underscorePasteRegex2 = /(?:^|\s)(_(?!\s+_)((?:[^_]+))_(?!\s+_))/g;
|
||
var Italic = Mark.create({
|
||
name: "italic",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "em"
|
||
},
|
||
{
|
||
tag: "i",
|
||
getAttrs: (node) => node.style.fontStyle !== "normal" && null
|
||
},
|
||
{
|
||
style: "font-style=normal",
|
||
clearMark: (mark) => mark.type.name === this.name
|
||
},
|
||
{
|
||
style: "font-style=italic"
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["em", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setItalic: () => ({ commands }) => {
|
||
return commands.setMark(this.name);
|
||
},
|
||
toggleItalic: () => ({ commands }) => {
|
||
return commands.toggleMark(this.name);
|
||
},
|
||
unsetItalic: () => ({ commands }) => {
|
||
return commands.unsetMark(this.name);
|
||
}
|
||
};
|
||
},
|
||
markdownTokenName: "em",
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.applyMark("italic", helpers.parseInline(token.tokens || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
return `*${h2.renderChildren(node)}*`;
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-i": () => this.editor.commands.toggleItalic(),
|
||
"Mod-I": () => this.editor.commands.toggleItalic()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
markInputRule({
|
||
find: starInputRegex2,
|
||
type: this.type
|
||
}),
|
||
markInputRule({
|
||
find: underscoreInputRegex2,
|
||
type: this.type
|
||
})
|
||
];
|
||
},
|
||
addPasteRules() {
|
||
return [
|
||
markPasteRule({
|
||
find: starPasteRegex2,
|
||
type: this.type
|
||
}),
|
||
markPasteRule({
|
||
find: underscorePasteRegex2,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-list/dist/index.js
|
||
var __defProp = Object.defineProperty;
|
||
var __export = (target, all) => {
|
||
for (var name in all)
|
||
__defProp(target, name, { get: all[name], enumerable: true });
|
||
};
|
||
var ListItemName = "listItem";
|
||
var TextStyleName = "textStyle";
|
||
var bulletListInputRegex = /^\s*([-+*])\s$/;
|
||
var BulletList = Node3.create({
|
||
name: "bulletList",
|
||
addOptions() {
|
||
return {
|
||
itemTypeName: "listItem",
|
||
HTMLAttributes: {},
|
||
keepMarks: false,
|
||
keepAttributes: false
|
||
};
|
||
},
|
||
group: "block list",
|
||
content() {
|
||
return `${this.options.itemTypeName}+`;
|
||
},
|
||
parseHTML() {
|
||
return [{ tag: "ul" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["ul", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
markdownTokenName: "list",
|
||
parseMarkdown: (token, helpers) => {
|
||
if (token.type !== "list" || token.ordered) {
|
||
return [];
|
||
}
|
||
return {
|
||
type: "bulletList",
|
||
content: token.items ? helpers.parseChildren(token.items) : []
|
||
};
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return h2.renderChildren(node.content, "\n");
|
||
},
|
||
markdownOptions: {
|
||
indentsContent: true
|
||
},
|
||
addCommands() {
|
||
return {
|
||
toggleBulletList: () => ({ commands, chain }) => {
|
||
if (this.options.keepAttributes) {
|
||
return chain().toggleList(this.name, this.options.itemTypeName, this.options.keepMarks).updateAttributes(ListItemName, this.editor.getAttributes(TextStyleName)).run();
|
||
}
|
||
return commands.toggleList(this.name, this.options.itemTypeName, this.options.keepMarks);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Shift-8": () => this.editor.commands.toggleBulletList()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
let inputRule = wrappingInputRule({
|
||
find: bulletListInputRegex,
|
||
type: this.type
|
||
});
|
||
if (this.options.keepMarks || this.options.keepAttributes) {
|
||
inputRule = wrappingInputRule({
|
||
find: bulletListInputRegex,
|
||
type: this.type,
|
||
keepMarks: this.options.keepMarks,
|
||
keepAttributes: this.options.keepAttributes,
|
||
getAttributes: () => {
|
||
return this.editor.getAttributes(TextStyleName);
|
||
},
|
||
editor: this.editor
|
||
});
|
||
}
|
||
return [inputRule];
|
||
}
|
||
});
|
||
var ListItem = Node3.create({
|
||
name: "listItem",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {},
|
||
bulletListTypeName: "bulletList",
|
||
orderedListTypeName: "orderedList"
|
||
};
|
||
},
|
||
content: "paragraph block*",
|
||
defining: true,
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "li"
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["li", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
markdownTokenName: "list_item",
|
||
parseMarkdown: (token, helpers) => {
|
||
if (token.type !== "list_item") {
|
||
return [];
|
||
}
|
||
let content = [];
|
||
if (token.tokens && token.tokens.length > 0) {
|
||
const hasParagraphTokens = token.tokens.some((t) => t.type === "paragraph");
|
||
if (hasParagraphTokens) {
|
||
content = helpers.parseChildren(token.tokens);
|
||
} else {
|
||
const firstToken = token.tokens[0];
|
||
if (firstToken && firstToken.type === "text" && firstToken.tokens && firstToken.tokens.length > 0) {
|
||
const inlineContent = helpers.parseInline(firstToken.tokens);
|
||
content = [
|
||
{
|
||
type: "paragraph",
|
||
content: inlineContent
|
||
}
|
||
];
|
||
if (token.tokens.length > 1) {
|
||
const remainingTokens = token.tokens.slice(1);
|
||
const additionalContent = helpers.parseChildren(remainingTokens);
|
||
content.push(...additionalContent);
|
||
}
|
||
} else {
|
||
content = helpers.parseChildren(token.tokens);
|
||
}
|
||
}
|
||
}
|
||
if (content.length === 0) {
|
||
content = [
|
||
{
|
||
type: "paragraph",
|
||
content: []
|
||
}
|
||
];
|
||
}
|
||
return {
|
||
type: "listItem",
|
||
content
|
||
};
|
||
},
|
||
renderMarkdown: (node, h2, ctx) => {
|
||
return renderNestedMarkdownContent(
|
||
node,
|
||
h2,
|
||
(context) => {
|
||
var _a, _b;
|
||
if (context.parentType === "bulletList") {
|
||
return "- ";
|
||
}
|
||
if (context.parentType === "orderedList") {
|
||
const start = ((_b = (_a = context.meta) == null ? void 0 : _a.parentAttrs) == null ? void 0 : _b.start) || 1;
|
||
return `${start + context.index}. `;
|
||
}
|
||
return "- ";
|
||
},
|
||
ctx
|
||
);
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
Enter: () => this.editor.commands.splitListItem(this.name),
|
||
Tab: () => this.editor.commands.sinkListItem(this.name),
|
||
"Shift-Tab": () => this.editor.commands.liftListItem(this.name)
|
||
};
|
||
}
|
||
});
|
||
var listHelpers_exports = {};
|
||
__export(listHelpers_exports, {
|
||
findListItemPos: () => findListItemPos,
|
||
getNextListDepth: () => getNextListDepth,
|
||
handleBackspace: () => handleBackspace,
|
||
handleDelete: () => handleDelete,
|
||
hasListBefore: () => hasListBefore,
|
||
hasListItemAfter: () => hasListItemAfter,
|
||
hasListItemBefore: () => hasListItemBefore,
|
||
listItemHasSubList: () => listItemHasSubList,
|
||
nextListIsDeeper: () => nextListIsDeeper,
|
||
nextListIsHigher: () => nextListIsHigher
|
||
});
|
||
var findListItemPos = (typeOrName, state) => {
|
||
const { $from } = state.selection;
|
||
const nodeType = getNodeType(typeOrName, state.schema);
|
||
let currentNode = null;
|
||
let currentDepth = $from.depth;
|
||
let currentPos = $from.pos;
|
||
let targetDepth = null;
|
||
while (currentDepth > 0 && targetDepth === null) {
|
||
currentNode = $from.node(currentDepth);
|
||
if (currentNode.type === nodeType) {
|
||
targetDepth = currentDepth;
|
||
} else {
|
||
currentDepth -= 1;
|
||
currentPos -= 1;
|
||
}
|
||
}
|
||
if (targetDepth === null) {
|
||
return null;
|
||
}
|
||
return { $pos: state.doc.resolve(currentPos), depth: targetDepth };
|
||
};
|
||
var getNextListDepth = (typeOrName, state) => {
|
||
const listItemPos = findListItemPos(typeOrName, state);
|
||
if (!listItemPos) {
|
||
return false;
|
||
}
|
||
const [, depth] = getNodeAtPosition(state, typeOrName, listItemPos.$pos.pos + 4);
|
||
return depth;
|
||
};
|
||
var hasListBefore = (editorState, name, parentListTypes) => {
|
||
const { $anchor } = editorState.selection;
|
||
const previousNodePos = Math.max(0, $anchor.pos - 2);
|
||
const previousNode = editorState.doc.resolve(previousNodePos).node();
|
||
if (!previousNode || !parentListTypes.includes(previousNode.type.name)) {
|
||
return false;
|
||
}
|
||
return true;
|
||
};
|
||
var hasListItemBefore = (typeOrName, state) => {
|
||
var _a;
|
||
const { $anchor } = state.selection;
|
||
const $targetPos = state.doc.resolve($anchor.pos - 2);
|
||
if ($targetPos.index() === 0) {
|
||
return false;
|
||
}
|
||
if (((_a = $targetPos.nodeBefore) == null ? void 0 : _a.type.name) !== typeOrName) {
|
||
return false;
|
||
}
|
||
return true;
|
||
};
|
||
var listItemHasSubList = (typeOrName, state, node) => {
|
||
if (!node) {
|
||
return false;
|
||
}
|
||
const nodeType = getNodeType(typeOrName, state.schema);
|
||
let hasSubList = false;
|
||
node.descendants((child) => {
|
||
if (child.type === nodeType) {
|
||
hasSubList = true;
|
||
}
|
||
});
|
||
return hasSubList;
|
||
};
|
||
var handleBackspace = (editor, name, parentListTypes) => {
|
||
if (editor.commands.undoInputRule()) {
|
||
return true;
|
||
}
|
||
if (editor.state.selection.from !== editor.state.selection.to) {
|
||
return false;
|
||
}
|
||
if (!isNodeActive(editor.state, name) && hasListBefore(editor.state, name, parentListTypes)) {
|
||
const { $anchor } = editor.state.selection;
|
||
const $listPos = editor.state.doc.resolve($anchor.before() - 1);
|
||
const listDescendants = [];
|
||
$listPos.node().descendants((node, pos) => {
|
||
if (node.type.name === name) {
|
||
listDescendants.push({ node, pos });
|
||
}
|
||
});
|
||
const lastItem = listDescendants.at(-1);
|
||
if (!lastItem) {
|
||
return false;
|
||
}
|
||
const $lastItemPos = editor.state.doc.resolve($listPos.start() + lastItem.pos + 1);
|
||
return editor.chain().cut({ from: $anchor.start() - 1, to: $anchor.end() + 1 }, $lastItemPos.end()).joinForward().run();
|
||
}
|
||
if (!isNodeActive(editor.state, name)) {
|
||
return false;
|
||
}
|
||
if (!isAtStartOfNode(editor.state)) {
|
||
return false;
|
||
}
|
||
const listItemPos = findListItemPos(name, editor.state);
|
||
if (!listItemPos) {
|
||
return false;
|
||
}
|
||
const $prev = editor.state.doc.resolve(listItemPos.$pos.pos - 2);
|
||
const prevNode = $prev.node(listItemPos.depth);
|
||
const previousListItemHasSubList = listItemHasSubList(name, editor.state, prevNode);
|
||
if (hasListItemBefore(name, editor.state) && !previousListItemHasSubList) {
|
||
return editor.commands.joinItemBackward();
|
||
}
|
||
return editor.chain().liftListItem(name).run();
|
||
};
|
||
var nextListIsDeeper = (typeOrName, state) => {
|
||
const listDepth = getNextListDepth(typeOrName, state);
|
||
const listItemPos = findListItemPos(typeOrName, state);
|
||
if (!listItemPos || !listDepth) {
|
||
return false;
|
||
}
|
||
if (listDepth > listItemPos.depth) {
|
||
return true;
|
||
}
|
||
return false;
|
||
};
|
||
var nextListIsHigher = (typeOrName, state) => {
|
||
const listDepth = getNextListDepth(typeOrName, state);
|
||
const listItemPos = findListItemPos(typeOrName, state);
|
||
if (!listItemPos || !listDepth) {
|
||
return false;
|
||
}
|
||
if (listDepth < listItemPos.depth) {
|
||
return true;
|
||
}
|
||
return false;
|
||
};
|
||
var handleDelete = (editor, name) => {
|
||
if (!isNodeActive(editor.state, name)) {
|
||
return false;
|
||
}
|
||
if (!isAtEndOfNode(editor.state, name)) {
|
||
return false;
|
||
}
|
||
const { selection } = editor.state;
|
||
const { $from, $to } = selection;
|
||
if (!selection.empty && $from.sameParent($to)) {
|
||
return false;
|
||
}
|
||
if (nextListIsDeeper(name, editor.state)) {
|
||
return editor.chain().focus(editor.state.selection.from + 4).lift(name).joinBackward().run();
|
||
}
|
||
if (nextListIsHigher(name, editor.state)) {
|
||
return editor.chain().joinForward().joinBackward().run();
|
||
}
|
||
return editor.commands.joinItemForward();
|
||
};
|
||
var hasListItemAfter = (typeOrName, state) => {
|
||
var _a;
|
||
const { $anchor } = state.selection;
|
||
const $targetPos = state.doc.resolve($anchor.pos - $anchor.parentOffset - 2);
|
||
if ($targetPos.index() === $targetPos.parent.childCount - 1) {
|
||
return false;
|
||
}
|
||
if (((_a = $targetPos.nodeAfter) == null ? void 0 : _a.type.name) !== typeOrName) {
|
||
return false;
|
||
}
|
||
return true;
|
||
};
|
||
var ListKeymap = Extension.create({
|
||
name: "listKeymap",
|
||
addOptions() {
|
||
return {
|
||
listTypes: [
|
||
{
|
||
itemName: "listItem",
|
||
wrapperNames: ["bulletList", "orderedList"]
|
||
},
|
||
{
|
||
itemName: "taskItem",
|
||
wrapperNames: ["taskList"]
|
||
}
|
||
]
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
Delete: ({ editor }) => {
|
||
let handled = false;
|
||
this.options.listTypes.forEach(({ itemName }) => {
|
||
if (editor.state.schema.nodes[itemName] === void 0) {
|
||
return;
|
||
}
|
||
if (handleDelete(editor, itemName)) {
|
||
handled = true;
|
||
}
|
||
});
|
||
return handled;
|
||
},
|
||
"Mod-Delete": ({ editor }) => {
|
||
let handled = false;
|
||
this.options.listTypes.forEach(({ itemName }) => {
|
||
if (editor.state.schema.nodes[itemName] === void 0) {
|
||
return;
|
||
}
|
||
if (handleDelete(editor, itemName)) {
|
||
handled = true;
|
||
}
|
||
});
|
||
return handled;
|
||
},
|
||
Backspace: ({ editor }) => {
|
||
let handled = false;
|
||
this.options.listTypes.forEach(({ itemName, wrapperNames }) => {
|
||
if (editor.state.schema.nodes[itemName] === void 0) {
|
||
return;
|
||
}
|
||
if (handleBackspace(editor, itemName, wrapperNames)) {
|
||
handled = true;
|
||
}
|
||
});
|
||
return handled;
|
||
},
|
||
"Mod-Backspace": ({ editor }) => {
|
||
let handled = false;
|
||
this.options.listTypes.forEach(({ itemName, wrapperNames }) => {
|
||
if (editor.state.schema.nodes[itemName] === void 0) {
|
||
return;
|
||
}
|
||
if (handleBackspace(editor, itemName, wrapperNames)) {
|
||
handled = true;
|
||
}
|
||
});
|
||
return handled;
|
||
}
|
||
};
|
||
}
|
||
});
|
||
var ORDERED_LIST_ITEM_REGEX = /^(\s*)(\d+)\.\s+(.*)$/;
|
||
var INDENTED_LINE_REGEX = /^\s/;
|
||
function collectOrderedListItems(lines) {
|
||
const listItems = [];
|
||
let currentLineIndex = 0;
|
||
let consumed = 0;
|
||
while (currentLineIndex < lines.length) {
|
||
const line = lines[currentLineIndex];
|
||
const match = line.match(ORDERED_LIST_ITEM_REGEX);
|
||
if (!match) {
|
||
break;
|
||
}
|
||
const [, indent, number, content] = match;
|
||
const indentLevel = indent.length;
|
||
let itemContent = content;
|
||
let nextLineIndex = currentLineIndex + 1;
|
||
const itemLines = [line];
|
||
while (nextLineIndex < lines.length) {
|
||
const nextLine = lines[nextLineIndex];
|
||
const nextMatch = nextLine.match(ORDERED_LIST_ITEM_REGEX);
|
||
if (nextMatch) {
|
||
break;
|
||
}
|
||
if (nextLine.trim() === "") {
|
||
itemLines.push(nextLine);
|
||
itemContent += "\n";
|
||
nextLineIndex += 1;
|
||
} else if (nextLine.match(INDENTED_LINE_REGEX)) {
|
||
itemLines.push(nextLine);
|
||
itemContent += `
|
||
${nextLine.slice(indentLevel + 2)}`;
|
||
nextLineIndex += 1;
|
||
} else {
|
||
break;
|
||
}
|
||
}
|
||
listItems.push({
|
||
indent: indentLevel,
|
||
number: parseInt(number, 10),
|
||
content: itemContent.trim(),
|
||
raw: itemLines.join("\n")
|
||
});
|
||
consumed = nextLineIndex;
|
||
currentLineIndex = nextLineIndex;
|
||
}
|
||
return [listItems, consumed];
|
||
}
|
||
function buildNestedStructure(items, baseIndent, lexer) {
|
||
var _a;
|
||
const result = [];
|
||
let currentIndex = 0;
|
||
while (currentIndex < items.length) {
|
||
const item = items[currentIndex];
|
||
if (item.indent === baseIndent) {
|
||
const contentLines = item.content.split("\n");
|
||
const mainText = ((_a = contentLines[0]) == null ? void 0 : _a.trim()) || "";
|
||
const tokens = [];
|
||
if (mainText) {
|
||
tokens.push({
|
||
type: "paragraph",
|
||
raw: mainText,
|
||
tokens: lexer.inlineTokens(mainText)
|
||
});
|
||
}
|
||
const additionalContent = contentLines.slice(1).join("\n").trim();
|
||
if (additionalContent) {
|
||
const blockTokens = lexer.blockTokens(additionalContent);
|
||
tokens.push(...blockTokens);
|
||
}
|
||
let lookAheadIndex = currentIndex + 1;
|
||
const nestedItems = [];
|
||
while (lookAheadIndex < items.length && items[lookAheadIndex].indent > baseIndent) {
|
||
nestedItems.push(items[lookAheadIndex]);
|
||
lookAheadIndex += 1;
|
||
}
|
||
if (nestedItems.length > 0) {
|
||
const nextIndent = Math.min(...nestedItems.map((nestedItem) => nestedItem.indent));
|
||
const nestedListItems = buildNestedStructure(nestedItems, nextIndent, lexer);
|
||
tokens.push({
|
||
type: "list",
|
||
ordered: true,
|
||
start: nestedItems[0].number,
|
||
items: nestedListItems,
|
||
raw: nestedItems.map((nestedItem) => nestedItem.raw).join("\n")
|
||
});
|
||
}
|
||
result.push({
|
||
type: "list_item",
|
||
raw: item.raw,
|
||
tokens
|
||
});
|
||
currentIndex = lookAheadIndex;
|
||
} else {
|
||
currentIndex += 1;
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
function parseListItems(items, helpers) {
|
||
return items.map((item) => {
|
||
if (item.type !== "list_item") {
|
||
return helpers.parseChildren([item])[0];
|
||
}
|
||
const content = [];
|
||
if (item.tokens && item.tokens.length > 0) {
|
||
item.tokens.forEach((itemToken) => {
|
||
if (itemToken.type === "paragraph" || itemToken.type === "list" || itemToken.type === "blockquote" || itemToken.type === "code") {
|
||
content.push(...helpers.parseChildren([itemToken]));
|
||
} else if (itemToken.type === "text" && itemToken.tokens) {
|
||
const inlineContent = helpers.parseChildren([itemToken]);
|
||
content.push({
|
||
type: "paragraph",
|
||
content: inlineContent
|
||
});
|
||
} else {
|
||
const parsed = helpers.parseChildren([itemToken]);
|
||
if (parsed.length > 0) {
|
||
content.push(...parsed);
|
||
}
|
||
}
|
||
});
|
||
}
|
||
return {
|
||
type: "listItem",
|
||
content
|
||
};
|
||
});
|
||
}
|
||
var ListItemName2 = "listItem";
|
||
var TextStyleName2 = "textStyle";
|
||
var orderedListInputRegex = /^(\d+)\.\s$/;
|
||
var OrderedList = Node3.create({
|
||
name: "orderedList",
|
||
addOptions() {
|
||
return {
|
||
itemTypeName: "listItem",
|
||
HTMLAttributes: {},
|
||
keepMarks: false,
|
||
keepAttributes: false
|
||
};
|
||
},
|
||
group: "block list",
|
||
content() {
|
||
return `${this.options.itemTypeName}+`;
|
||
},
|
||
addAttributes() {
|
||
return {
|
||
start: {
|
||
default: 1,
|
||
parseHTML: (element) => {
|
||
return element.hasAttribute("start") ? parseInt(element.getAttribute("start") || "", 10) : 1;
|
||
}
|
||
},
|
||
type: {
|
||
default: null,
|
||
parseHTML: (element) => element.getAttribute("type")
|
||
}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "ol"
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
const { start, ...attributesWithoutStart } = HTMLAttributes;
|
||
return start === 1 ? ["ol", mergeAttributes(this.options.HTMLAttributes, attributesWithoutStart), 0] : ["ol", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
markdownTokenName: "list",
|
||
parseMarkdown: (token, helpers) => {
|
||
if (token.type !== "list" || !token.ordered) {
|
||
return [];
|
||
}
|
||
const startValue = token.start || 1;
|
||
const content = token.items ? parseListItems(token.items, helpers) : [];
|
||
if (startValue !== 1) {
|
||
return {
|
||
type: "orderedList",
|
||
attrs: { start: startValue },
|
||
content
|
||
};
|
||
}
|
||
return {
|
||
type: "orderedList",
|
||
content
|
||
};
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return h2.renderChildren(node.content, "\n");
|
||
},
|
||
markdownTokenizer: {
|
||
name: "orderedList",
|
||
level: "block",
|
||
start: (src) => {
|
||
const match = src.match(/^(\s*)(\d+)\.\s+/);
|
||
const index = match == null ? void 0 : match.index;
|
||
return index !== void 0 ? index : -1;
|
||
},
|
||
tokenize: (src, _tokens, lexer) => {
|
||
var _a;
|
||
const lines = src.split("\n");
|
||
const [listItems, consumed] = collectOrderedListItems(lines);
|
||
if (listItems.length === 0) {
|
||
return void 0;
|
||
}
|
||
const items = buildNestedStructure(listItems, 0, lexer);
|
||
if (items.length === 0) {
|
||
return void 0;
|
||
}
|
||
const startValue = ((_a = listItems[0]) == null ? void 0 : _a.number) || 1;
|
||
return {
|
||
type: "list",
|
||
ordered: true,
|
||
start: startValue,
|
||
items,
|
||
raw: lines.slice(0, consumed).join("\n")
|
||
};
|
||
}
|
||
},
|
||
markdownOptions: {
|
||
indentsContent: true
|
||
},
|
||
addCommands() {
|
||
return {
|
||
toggleOrderedList: () => ({ commands, chain }) => {
|
||
if (this.options.keepAttributes) {
|
||
return chain().toggleList(this.name, this.options.itemTypeName, this.options.keepMarks).updateAttributes(ListItemName2, this.editor.getAttributes(TextStyleName2)).run();
|
||
}
|
||
return commands.toggleList(this.name, this.options.itemTypeName, this.options.keepMarks);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Shift-7": () => this.editor.commands.toggleOrderedList()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
let inputRule = wrappingInputRule({
|
||
find: orderedListInputRegex,
|
||
type: this.type,
|
||
getAttributes: (match) => ({ start: +match[1] }),
|
||
joinPredicate: (match, node) => node.childCount + node.attrs.start === +match[1]
|
||
});
|
||
if (this.options.keepMarks || this.options.keepAttributes) {
|
||
inputRule = wrappingInputRule({
|
||
find: orderedListInputRegex,
|
||
type: this.type,
|
||
keepMarks: this.options.keepMarks,
|
||
keepAttributes: this.options.keepAttributes,
|
||
getAttributes: (match) => ({ start: +match[1], ...this.editor.getAttributes(TextStyleName2) }),
|
||
joinPredicate: (match, node) => node.childCount + node.attrs.start === +match[1],
|
||
editor: this.editor
|
||
});
|
||
}
|
||
return [inputRule];
|
||
}
|
||
});
|
||
var inputRegex3 = /^\s*(\[([( |x])?\])\s$/;
|
||
var TaskItem = Node3.create({
|
||
name: "taskItem",
|
||
addOptions() {
|
||
return {
|
||
nested: false,
|
||
HTMLAttributes: {},
|
||
taskListTypeName: "taskList",
|
||
a11y: void 0
|
||
};
|
||
},
|
||
content() {
|
||
return this.options.nested ? "paragraph block*" : "paragraph+";
|
||
},
|
||
defining: true,
|
||
addAttributes() {
|
||
return {
|
||
checked: {
|
||
default: false,
|
||
keepOnSplit: false,
|
||
parseHTML: (element) => {
|
||
const dataChecked = element.getAttribute("data-checked");
|
||
return dataChecked === "" || dataChecked === "true";
|
||
},
|
||
renderHTML: (attributes) => ({
|
||
"data-checked": attributes.checked
|
||
})
|
||
}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: `li[data-type="${this.name}"]`,
|
||
priority: 51
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ node, HTMLAttributes }) {
|
||
return [
|
||
"li",
|
||
mergeAttributes(this.options.HTMLAttributes, HTMLAttributes, {
|
||
"data-type": this.name
|
||
}),
|
||
[
|
||
"label",
|
||
[
|
||
"input",
|
||
{
|
||
type: "checkbox",
|
||
checked: node.attrs.checked ? "checked" : null
|
||
}
|
||
],
|
||
["span"]
|
||
],
|
||
["div", 0]
|
||
];
|
||
},
|
||
parseMarkdown: (token, h2) => {
|
||
const content = [];
|
||
if (token.tokens && token.tokens.length > 0) {
|
||
content.push(h2.createNode("paragraph", {}, h2.parseInline(token.tokens)));
|
||
} else if (token.text) {
|
||
content.push(h2.createNode("paragraph", {}, [h2.createNode("text", { text: token.text })]));
|
||
} else {
|
||
content.push(h2.createNode("paragraph", {}, []));
|
||
}
|
||
if (token.nestedTokens && token.nestedTokens.length > 0) {
|
||
const nestedContent = h2.parseChildren(token.nestedTokens);
|
||
content.push(...nestedContent);
|
||
}
|
||
return h2.createNode("taskItem", { checked: token.checked || false }, content);
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
var _a;
|
||
const checkedChar = ((_a = node.attrs) == null ? void 0 : _a.checked) ? "x" : " ";
|
||
const prefix = `- [${checkedChar}] `;
|
||
return renderNestedMarkdownContent(node, h2, prefix);
|
||
},
|
||
addKeyboardShortcuts() {
|
||
const shortcuts = {
|
||
Enter: () => this.editor.commands.splitListItem(this.name),
|
||
"Shift-Tab": () => this.editor.commands.liftListItem(this.name)
|
||
};
|
||
if (!this.options.nested) {
|
||
return shortcuts;
|
||
}
|
||
return {
|
||
...shortcuts,
|
||
Tab: () => this.editor.commands.sinkListItem(this.name)
|
||
};
|
||
},
|
||
addNodeView() {
|
||
return ({ node, HTMLAttributes, getPos, editor }) => {
|
||
const listItem = document.createElement("li");
|
||
const checkboxWrapper = document.createElement("label");
|
||
const checkboxStyler = document.createElement("span");
|
||
const checkbox = document.createElement("input");
|
||
const content = document.createElement("div");
|
||
const updateA11Y = (currentNode) => {
|
||
var _a, _b;
|
||
checkbox.ariaLabel = ((_b = (_a = this.options.a11y) == null ? void 0 : _a.checkboxLabel) == null ? void 0 : _b.call(_a, currentNode, checkbox.checked)) || `Task item checkbox for ${currentNode.textContent || "empty task item"}`;
|
||
};
|
||
updateA11Y(node);
|
||
checkboxWrapper.contentEditable = "false";
|
||
checkbox.type = "checkbox";
|
||
checkbox.addEventListener("mousedown", (event) => event.preventDefault());
|
||
checkbox.addEventListener("change", (event) => {
|
||
if (!editor.isEditable && !this.options.onReadOnlyChecked) {
|
||
checkbox.checked = !checkbox.checked;
|
||
return;
|
||
}
|
||
const { checked } = event.target;
|
||
if (editor.isEditable && typeof getPos === "function") {
|
||
editor.chain().focus(void 0, { scrollIntoView: false }).command(({ tr }) => {
|
||
const position = getPos();
|
||
if (typeof position !== "number") {
|
||
return false;
|
||
}
|
||
const currentNode = tr.doc.nodeAt(position);
|
||
tr.setNodeMarkup(position, void 0, {
|
||
...currentNode == null ? void 0 : currentNode.attrs,
|
||
checked
|
||
});
|
||
return true;
|
||
}).run();
|
||
}
|
||
if (!editor.isEditable && this.options.onReadOnlyChecked) {
|
||
if (!this.options.onReadOnlyChecked(node, checked)) {
|
||
checkbox.checked = !checkbox.checked;
|
||
}
|
||
}
|
||
});
|
||
Object.entries(this.options.HTMLAttributes).forEach(([key, value]) => {
|
||
listItem.setAttribute(key, value);
|
||
});
|
||
listItem.dataset.checked = node.attrs.checked;
|
||
checkbox.checked = node.attrs.checked;
|
||
checkboxWrapper.append(checkbox, checkboxStyler);
|
||
listItem.append(checkboxWrapper, content);
|
||
Object.entries(HTMLAttributes).forEach(([key, value]) => {
|
||
listItem.setAttribute(key, value);
|
||
});
|
||
let prevRenderedAttributeKeys = new Set(Object.keys(HTMLAttributes));
|
||
return {
|
||
dom: listItem,
|
||
contentDOM: content,
|
||
update: (updatedNode) => {
|
||
if (updatedNode.type !== this.type) {
|
||
return false;
|
||
}
|
||
listItem.dataset.checked = updatedNode.attrs.checked;
|
||
checkbox.checked = updatedNode.attrs.checked;
|
||
updateA11Y(updatedNode);
|
||
const extensionAttributes = editor.extensionManager.attributes;
|
||
const newHTMLAttributes = getRenderedAttributes(updatedNode, extensionAttributes);
|
||
const newKeys = new Set(Object.keys(newHTMLAttributes));
|
||
const staticAttrs = this.options.HTMLAttributes;
|
||
prevRenderedAttributeKeys.forEach((key) => {
|
||
if (!newKeys.has(key)) {
|
||
if (key in staticAttrs) {
|
||
listItem.setAttribute(key, staticAttrs[key]);
|
||
} else {
|
||
listItem.removeAttribute(key);
|
||
}
|
||
}
|
||
});
|
||
Object.entries(newHTMLAttributes).forEach(([key, value]) => {
|
||
if (value === null || value === void 0) {
|
||
if (key in staticAttrs) {
|
||
listItem.setAttribute(key, staticAttrs[key]);
|
||
} else {
|
||
listItem.removeAttribute(key);
|
||
}
|
||
} else {
|
||
listItem.setAttribute(key, value);
|
||
}
|
||
});
|
||
prevRenderedAttributeKeys = newKeys;
|
||
return true;
|
||
}
|
||
};
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
wrappingInputRule({
|
||
find: inputRegex3,
|
||
type: this.type,
|
||
getAttributes: (match) => ({
|
||
checked: match[match.length - 1] === "x"
|
||
})
|
||
})
|
||
];
|
||
}
|
||
});
|
||
var TaskList = Node3.create({
|
||
name: "taskList",
|
||
addOptions() {
|
||
return {
|
||
itemTypeName: "taskItem",
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
group: "block list",
|
||
content() {
|
||
return `${this.options.itemTypeName}+`;
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: `ul[data-type="${this.name}"]`,
|
||
priority: 51
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["ul", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes, { "data-type": this.name }), 0];
|
||
},
|
||
parseMarkdown: (token, h2) => {
|
||
return h2.createNode("taskList", {}, h2.parseChildren(token.items || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node.content) {
|
||
return "";
|
||
}
|
||
return h2.renderChildren(node.content, "\n");
|
||
},
|
||
markdownTokenizer: {
|
||
name: "taskList",
|
||
level: "block",
|
||
start(src) {
|
||
var _a;
|
||
const index = (_a = src.match(/^\s*[-+*]\s+\[([ xX])\]\s+/)) == null ? void 0 : _a.index;
|
||
return index !== void 0 ? index : -1;
|
||
},
|
||
tokenize(src, tokens, lexer) {
|
||
const parseTaskListContent = (content) => {
|
||
const nestedResult = parseIndentedBlocks(
|
||
content,
|
||
{
|
||
itemPattern: /^(\s*)([-+*])\s+\[([ xX])\]\s+(.*)$/,
|
||
extractItemData: (match) => ({
|
||
indentLevel: match[1].length,
|
||
mainContent: match[4],
|
||
checked: match[3].toLowerCase() === "x"
|
||
}),
|
||
createToken: (data, nestedTokens) => ({
|
||
type: "taskItem",
|
||
raw: "",
|
||
mainContent: data.mainContent,
|
||
indentLevel: data.indentLevel,
|
||
checked: data.checked,
|
||
text: data.mainContent,
|
||
tokens: lexer.inlineTokens(data.mainContent),
|
||
nestedTokens
|
||
}),
|
||
// Allow recursive nesting
|
||
customNestedParser: parseTaskListContent
|
||
},
|
||
lexer
|
||
);
|
||
if (nestedResult) {
|
||
return [
|
||
{
|
||
type: "taskList",
|
||
raw: nestedResult.raw,
|
||
items: nestedResult.items
|
||
}
|
||
];
|
||
}
|
||
return lexer.blockTokens(content);
|
||
};
|
||
const result = parseIndentedBlocks(
|
||
src,
|
||
{
|
||
itemPattern: /^(\s*)([-+*])\s+\[([ xX])\]\s+(.*)$/,
|
||
extractItemData: (match) => ({
|
||
indentLevel: match[1].length,
|
||
mainContent: match[4],
|
||
checked: match[3].toLowerCase() === "x"
|
||
}),
|
||
createToken: (data, nestedTokens) => ({
|
||
type: "taskItem",
|
||
raw: "",
|
||
mainContent: data.mainContent,
|
||
indentLevel: data.indentLevel,
|
||
checked: data.checked,
|
||
text: data.mainContent,
|
||
tokens: lexer.inlineTokens(data.mainContent),
|
||
nestedTokens
|
||
}),
|
||
// Use the recursive parser for nested content
|
||
customNestedParser: parseTaskListContent
|
||
},
|
||
lexer
|
||
);
|
||
if (!result) {
|
||
return void 0;
|
||
}
|
||
return {
|
||
type: "taskList",
|
||
raw: result.raw,
|
||
items: result.items
|
||
};
|
||
}
|
||
},
|
||
markdownOptions: {
|
||
indentsContent: true
|
||
},
|
||
addCommands() {
|
||
return {
|
||
toggleTaskList: () => ({ commands }) => {
|
||
return commands.toggleList(this.name, this.options.itemTypeName);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Shift-9": () => this.editor.commands.toggleTaskList()
|
||
};
|
||
}
|
||
});
|
||
var ListKit = Extension.create({
|
||
name: "listKit",
|
||
addExtensions() {
|
||
const extensions = [];
|
||
if (this.options.bulletList !== false) {
|
||
extensions.push(BulletList.configure(this.options.bulletList));
|
||
}
|
||
if (this.options.listItem !== false) {
|
||
extensions.push(ListItem.configure(this.options.listItem));
|
||
}
|
||
if (this.options.listKeymap !== false) {
|
||
extensions.push(ListKeymap.configure(this.options.listKeymap));
|
||
}
|
||
if (this.options.orderedList !== false) {
|
||
extensions.push(OrderedList.configure(this.options.orderedList));
|
||
}
|
||
if (this.options.taskItem !== false) {
|
||
extensions.push(TaskItem.configure(this.options.taskItem));
|
||
}
|
||
if (this.options.taskList !== false) {
|
||
extensions.push(TaskList.configure(this.options.taskList));
|
||
}
|
||
return extensions;
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-paragraph/dist/index.js
|
||
var EMPTY_PARAGRAPH_MARKDOWN = " ";
|
||
var NBSP_CHAR = " ";
|
||
var Paragraph = Node3.create({
|
||
name: "paragraph",
|
||
priority: 1e3,
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
group: "block",
|
||
content: "inline*",
|
||
parseHTML() {
|
||
return [{ tag: "p" }];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["p", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
parseMarkdown: (token, helpers) => {
|
||
const tokens = token.tokens || [];
|
||
if (tokens.length === 1 && tokens[0].type === "image") {
|
||
return helpers.parseChildren([tokens[0]]);
|
||
}
|
||
const content = helpers.parseInline(tokens);
|
||
if (content.length === 1 && content[0].type === "text" && (content[0].text === EMPTY_PARAGRAPH_MARKDOWN || content[0].text === NBSP_CHAR)) {
|
||
return helpers.createNode("paragraph", void 0, []);
|
||
}
|
||
return helpers.createNode("paragraph", void 0, content);
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
if (!node) {
|
||
return "";
|
||
}
|
||
const content = Array.isArray(node.content) ? node.content : [];
|
||
if (content.length === 0) {
|
||
return EMPTY_PARAGRAPH_MARKDOWN;
|
||
}
|
||
return h2.renderChildren(content);
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setParagraph: () => ({ commands }) => {
|
||
return commands.setNode(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Alt-0": () => this.editor.commands.setParagraph()
|
||
};
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-strike/dist/index.js
|
||
var inputRegex4 = /(?:^|\s)(~~(?!\s+~~)((?:[^~]+))~~(?!\s+~~))$/;
|
||
var pasteRegex2 = /(?:^|\s)(~~(?!\s+~~)((?:[^~]+))~~(?!\s+~~))/g;
|
||
var Strike = Mark.create({
|
||
name: "strike",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "s"
|
||
},
|
||
{
|
||
tag: "del"
|
||
},
|
||
{
|
||
tag: "strike"
|
||
},
|
||
{
|
||
style: "text-decoration",
|
||
consuming: false,
|
||
getAttrs: (style) => style.includes("line-through") ? {} : false
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["s", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
markdownTokenName: "del",
|
||
parseMarkdown: (token, helpers) => {
|
||
return helpers.applyMark("strike", helpers.parseInline(token.tokens || []));
|
||
},
|
||
renderMarkdown: (node, h2) => {
|
||
return `~~${h2.renderChildren(node)}~~`;
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setStrike: () => ({ commands }) => {
|
||
return commands.setMark(this.name);
|
||
},
|
||
toggleStrike: () => ({ commands }) => {
|
||
return commands.toggleMark(this.name);
|
||
},
|
||
unsetStrike: () => ({ commands }) => {
|
||
return commands.unsetMark(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-Shift-s": () => this.editor.commands.toggleStrike()
|
||
};
|
||
},
|
||
addInputRules() {
|
||
return [
|
||
markInputRule({
|
||
find: inputRegex4,
|
||
type: this.type
|
||
})
|
||
];
|
||
},
|
||
addPasteRules() {
|
||
return [
|
||
markPasteRule({
|
||
find: pasteRegex2,
|
||
type: this.type
|
||
})
|
||
];
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-text/dist/index.js
|
||
var Text = Node3.create({
|
||
name: "text",
|
||
group: "inline",
|
||
parseMarkdown: (token) => {
|
||
return {
|
||
type: "text",
|
||
text: token.text || ""
|
||
};
|
||
},
|
||
renderMarkdown: (node) => node.text || ""
|
||
});
|
||
|
||
// node_modules/@tiptap/extension-underline/dist/index.js
|
||
var Underline = Mark.create({
|
||
name: "underline",
|
||
addOptions() {
|
||
return {
|
||
HTMLAttributes: {}
|
||
};
|
||
},
|
||
parseHTML() {
|
||
return [
|
||
{
|
||
tag: "u"
|
||
},
|
||
{
|
||
style: "text-decoration",
|
||
consuming: false,
|
||
getAttrs: (style) => style.includes("underline") ? {} : false
|
||
}
|
||
];
|
||
},
|
||
renderHTML({ HTMLAttributes }) {
|
||
return ["u", mergeAttributes(this.options.HTMLAttributes, HTMLAttributes), 0];
|
||
},
|
||
parseMarkdown(token, helpers) {
|
||
return helpers.applyMark(this.name || "underline", helpers.parseInline(token.tokens || []));
|
||
},
|
||
renderMarkdown(node, helpers) {
|
||
return `++${helpers.renderChildren(node)}++`;
|
||
},
|
||
markdownTokenizer: {
|
||
name: "underline",
|
||
level: "inline",
|
||
start(src) {
|
||
return src.indexOf("++");
|
||
},
|
||
tokenize(src, _tokens, lexer) {
|
||
const rule = /^(\+\+)([\s\S]+?)(\+\+)/;
|
||
const match = rule.exec(src);
|
||
if (!match) {
|
||
return void 0;
|
||
}
|
||
const innerContent = match[2].trim();
|
||
return {
|
||
type: "underline",
|
||
raw: match[0],
|
||
text: innerContent,
|
||
tokens: lexer.inlineTokens(innerContent)
|
||
};
|
||
}
|
||
},
|
||
addCommands() {
|
||
return {
|
||
setUnderline: () => ({ commands }) => {
|
||
return commands.setMark(this.name);
|
||
},
|
||
toggleUnderline: () => ({ commands }) => {
|
||
return commands.toggleMark(this.name);
|
||
},
|
||
unsetUnderline: () => ({ commands }) => {
|
||
return commands.unsetMark(this.name);
|
||
}
|
||
};
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-u": () => this.editor.commands.toggleUnderline(),
|
||
"Mod-U": () => this.editor.commands.toggleUnderline()
|
||
};
|
||
}
|
||
});
|
||
|
||
// node_modules/prosemirror-dropcursor/dist/index.js
|
||
function dropCursor(options = {}) {
|
||
return new Plugin({
|
||
view(editorView) {
|
||
return new DropCursorView(editorView, options);
|
||
}
|
||
});
|
||
}
|
||
var DropCursorView = class {
|
||
constructor(editorView, options) {
|
||
var _a;
|
||
this.editorView = editorView;
|
||
this.cursorPos = null;
|
||
this.element = null;
|
||
this.timeout = -1;
|
||
this.width = (_a = options.width) !== null && _a !== void 0 ? _a : 1;
|
||
this.color = options.color === false ? void 0 : options.color || "black";
|
||
this.class = options.class;
|
||
this.handlers = ["dragover", "dragend", "drop", "dragleave"].map((name) => {
|
||
let handler = (e) => {
|
||
this[name](e);
|
||
};
|
||
editorView.dom.addEventListener(name, handler);
|
||
return { name, handler };
|
||
});
|
||
}
|
||
destroy() {
|
||
this.handlers.forEach(({ name, handler }) => this.editorView.dom.removeEventListener(name, handler));
|
||
}
|
||
update(editorView, prevState) {
|
||
if (this.cursorPos != null && prevState.doc != editorView.state.doc) {
|
||
if (this.cursorPos > editorView.state.doc.content.size)
|
||
this.setCursor(null);
|
||
else
|
||
this.updateOverlay();
|
||
}
|
||
}
|
||
setCursor(pos) {
|
||
if (pos == this.cursorPos)
|
||
return;
|
||
this.cursorPos = pos;
|
||
if (pos == null) {
|
||
this.element.parentNode.removeChild(this.element);
|
||
this.element = null;
|
||
} else {
|
||
this.updateOverlay();
|
||
}
|
||
}
|
||
updateOverlay() {
|
||
let $pos = this.editorView.state.doc.resolve(this.cursorPos);
|
||
let isBlock = !$pos.parent.inlineContent, rect;
|
||
let editorDOM = this.editorView.dom, editorRect = editorDOM.getBoundingClientRect();
|
||
let scaleX = editorRect.width / editorDOM.offsetWidth, scaleY = editorRect.height / editorDOM.offsetHeight;
|
||
if (isBlock) {
|
||
let before = $pos.nodeBefore, after = $pos.nodeAfter;
|
||
if (before || after) {
|
||
let node = this.editorView.nodeDOM(this.cursorPos - (before ? before.nodeSize : 0));
|
||
if (node) {
|
||
let nodeRect = node.getBoundingClientRect();
|
||
let top = before ? nodeRect.bottom : nodeRect.top;
|
||
if (before && after)
|
||
top = (top + this.editorView.nodeDOM(this.cursorPos).getBoundingClientRect().top) / 2;
|
||
let halfWidth = this.width / 2 * scaleY;
|
||
rect = { left: nodeRect.left, right: nodeRect.right, top: top - halfWidth, bottom: top + halfWidth };
|
||
}
|
||
}
|
||
}
|
||
if (!rect) {
|
||
let coords = this.editorView.coordsAtPos(this.cursorPos);
|
||
let halfWidth = this.width / 2 * scaleX;
|
||
rect = { left: coords.left - halfWidth, right: coords.left + halfWidth, top: coords.top, bottom: coords.bottom };
|
||
}
|
||
let parent = this.editorView.dom.offsetParent;
|
||
if (!this.element) {
|
||
this.element = parent.appendChild(document.createElement("div"));
|
||
if (this.class)
|
||
this.element.className = this.class;
|
||
this.element.style.cssText = "position: absolute; z-index: 50; pointer-events: none;";
|
||
if (this.color) {
|
||
this.element.style.backgroundColor = this.color;
|
||
}
|
||
}
|
||
this.element.classList.toggle("prosemirror-dropcursor-block", isBlock);
|
||
this.element.classList.toggle("prosemirror-dropcursor-inline", !isBlock);
|
||
let parentLeft, parentTop;
|
||
if (!parent || parent == document.body && getComputedStyle(parent).position == "static") {
|
||
parentLeft = -pageXOffset;
|
||
parentTop = -pageYOffset;
|
||
} else {
|
||
let rect2 = parent.getBoundingClientRect();
|
||
let parentScaleX = rect2.width / parent.offsetWidth, parentScaleY = rect2.height / parent.offsetHeight;
|
||
parentLeft = rect2.left - parent.scrollLeft * parentScaleX;
|
||
parentTop = rect2.top - parent.scrollTop * parentScaleY;
|
||
}
|
||
this.element.style.left = (rect.left - parentLeft) / scaleX + "px";
|
||
this.element.style.top = (rect.top - parentTop) / scaleY + "px";
|
||
this.element.style.width = (rect.right - rect.left) / scaleX + "px";
|
||
this.element.style.height = (rect.bottom - rect.top) / scaleY + "px";
|
||
}
|
||
scheduleRemoval(timeout) {
|
||
clearTimeout(this.timeout);
|
||
this.timeout = setTimeout(() => this.setCursor(null), timeout);
|
||
}
|
||
dragover(event) {
|
||
if (!this.editorView.editable)
|
||
return;
|
||
let pos = this.editorView.posAtCoords({ left: event.clientX, top: event.clientY });
|
||
let node = pos && pos.inside >= 0 && this.editorView.state.doc.nodeAt(pos.inside);
|
||
let disableDropCursor = node && node.type.spec.disableDropCursor;
|
||
let disabled = typeof disableDropCursor == "function" ? disableDropCursor(this.editorView, pos, event) : disableDropCursor;
|
||
if (pos && !disabled) {
|
||
let target = pos.pos;
|
||
if (this.editorView.dragging && this.editorView.dragging.slice) {
|
||
let point = dropPoint(this.editorView.state.doc, target, this.editorView.dragging.slice);
|
||
if (point != null)
|
||
target = point;
|
||
}
|
||
this.setCursor(target);
|
||
this.scheduleRemoval(5e3);
|
||
}
|
||
}
|
||
dragend() {
|
||
this.scheduleRemoval(20);
|
||
}
|
||
drop() {
|
||
this.scheduleRemoval(20);
|
||
}
|
||
dragleave(event) {
|
||
if (!this.editorView.dom.contains(event.relatedTarget))
|
||
this.setCursor(null);
|
||
}
|
||
};
|
||
|
||
// node_modules/prosemirror-gapcursor/dist/index.js
|
||
var GapCursor = class _GapCursor extends Selection {
|
||
/**
|
||
Create a gap cursor.
|
||
*/
|
||
constructor($pos) {
|
||
super($pos, $pos);
|
||
}
|
||
map(doc, mapping) {
|
||
let $pos = doc.resolve(mapping.map(this.head));
|
||
return _GapCursor.valid($pos) ? new _GapCursor($pos) : Selection.near($pos);
|
||
}
|
||
content() {
|
||
return Slice.empty;
|
||
}
|
||
eq(other) {
|
||
return other instanceof _GapCursor && other.head == this.head;
|
||
}
|
||
toJSON() {
|
||
return { type: "gapcursor", pos: this.head };
|
||
}
|
||
/**
|
||
@internal
|
||
*/
|
||
static fromJSON(doc, json) {
|
||
if (typeof json.pos != "number")
|
||
throw new RangeError("Invalid input for GapCursor.fromJSON");
|
||
return new _GapCursor(doc.resolve(json.pos));
|
||
}
|
||
/**
|
||
@internal
|
||
*/
|
||
getBookmark() {
|
||
return new GapBookmark(this.anchor);
|
||
}
|
||
/**
|
||
@internal
|
||
*/
|
||
static valid($pos) {
|
||
let parent = $pos.parent;
|
||
if (parent.isTextblock || !closedBefore($pos) || !closedAfter($pos))
|
||
return false;
|
||
let override = parent.type.spec.allowGapCursor;
|
||
if (override != null)
|
||
return override;
|
||
let deflt = parent.contentMatchAt($pos.index()).defaultType;
|
||
return deflt && deflt.isTextblock;
|
||
}
|
||
/**
|
||
@internal
|
||
*/
|
||
static findGapCursorFrom($pos, dir, mustMove = false) {
|
||
search: for (; ; ) {
|
||
if (!mustMove && _GapCursor.valid($pos))
|
||
return $pos;
|
||
let pos = $pos.pos, next = null;
|
||
for (let d = $pos.depth; ; d--) {
|
||
let parent = $pos.node(d);
|
||
if (dir > 0 ? $pos.indexAfter(d) < parent.childCount : $pos.index(d) > 0) {
|
||
next = parent.child(dir > 0 ? $pos.indexAfter(d) : $pos.index(d) - 1);
|
||
break;
|
||
} else if (d == 0) {
|
||
return null;
|
||
}
|
||
pos += dir;
|
||
let $cur = $pos.doc.resolve(pos);
|
||
if (_GapCursor.valid($cur))
|
||
return $cur;
|
||
}
|
||
for (; ; ) {
|
||
let inside = dir > 0 ? next.firstChild : next.lastChild;
|
||
if (!inside) {
|
||
if (next.isAtom && !next.isText && !NodeSelection.isSelectable(next)) {
|
||
$pos = $pos.doc.resolve(pos + next.nodeSize * dir);
|
||
mustMove = false;
|
||
continue search;
|
||
}
|
||
break;
|
||
}
|
||
next = inside;
|
||
pos += dir;
|
||
let $cur = $pos.doc.resolve(pos);
|
||
if (_GapCursor.valid($cur))
|
||
return $cur;
|
||
}
|
||
return null;
|
||
}
|
||
}
|
||
};
|
||
GapCursor.prototype.visible = false;
|
||
GapCursor.findFrom = GapCursor.findGapCursorFrom;
|
||
Selection.jsonID("gapcursor", GapCursor);
|
||
var GapBookmark = class _GapBookmark {
|
||
constructor(pos) {
|
||
this.pos = pos;
|
||
}
|
||
map(mapping) {
|
||
return new _GapBookmark(mapping.map(this.pos));
|
||
}
|
||
resolve(doc) {
|
||
let $pos = doc.resolve(this.pos);
|
||
return GapCursor.valid($pos) ? new GapCursor($pos) : Selection.near($pos);
|
||
}
|
||
};
|
||
function needsGap(type) {
|
||
return type.isAtom || type.spec.isolating || type.spec.createGapCursor;
|
||
}
|
||
function closedBefore($pos) {
|
||
for (let d = $pos.depth; d >= 0; d--) {
|
||
let index = $pos.index(d), parent = $pos.node(d);
|
||
if (index == 0) {
|
||
if (parent.type.spec.isolating)
|
||
return true;
|
||
continue;
|
||
}
|
||
for (let before = parent.child(index - 1); ; before = before.lastChild) {
|
||
if (before.childCount == 0 && !before.inlineContent || needsGap(before.type))
|
||
return true;
|
||
if (before.inlineContent)
|
||
return false;
|
||
}
|
||
}
|
||
return true;
|
||
}
|
||
function closedAfter($pos) {
|
||
for (let d = $pos.depth; d >= 0; d--) {
|
||
let index = $pos.indexAfter(d), parent = $pos.node(d);
|
||
if (index == parent.childCount) {
|
||
if (parent.type.spec.isolating)
|
||
return true;
|
||
continue;
|
||
}
|
||
for (let after = parent.child(index); ; after = after.firstChild) {
|
||
if (after.childCount == 0 && !after.inlineContent || needsGap(after.type))
|
||
return true;
|
||
if (after.inlineContent)
|
||
return false;
|
||
}
|
||
}
|
||
return true;
|
||
}
|
||
function gapCursor() {
|
||
return new Plugin({
|
||
props: {
|
||
decorations: drawGapCursor,
|
||
createSelectionBetween(_view, $anchor, $head) {
|
||
return $anchor.pos == $head.pos && GapCursor.valid($head) ? new GapCursor($head) : null;
|
||
},
|
||
handleClick,
|
||
handleKeyDown,
|
||
handleDOMEvents: { beforeinput }
|
||
}
|
||
});
|
||
}
|
||
var handleKeyDown = keydownHandler({
|
||
"ArrowLeft": arrow("horiz", -1),
|
||
"ArrowRight": arrow("horiz", 1),
|
||
"ArrowUp": arrow("vert", -1),
|
||
"ArrowDown": arrow("vert", 1)
|
||
});
|
||
function arrow(axis, dir) {
|
||
const dirStr = axis == "vert" ? dir > 0 ? "down" : "up" : dir > 0 ? "right" : "left";
|
||
return function(state, dispatch, view) {
|
||
let sel = state.selection;
|
||
let $start = dir > 0 ? sel.$to : sel.$from, mustMove = sel.empty;
|
||
if (sel instanceof TextSelection) {
|
||
if (!view.endOfTextblock(dirStr) || $start.depth == 0)
|
||
return false;
|
||
mustMove = false;
|
||
$start = state.doc.resolve(dir > 0 ? $start.after() : $start.before());
|
||
}
|
||
let $found = GapCursor.findGapCursorFrom($start, dir, mustMove);
|
||
if (!$found)
|
||
return false;
|
||
if (dispatch)
|
||
dispatch(state.tr.setSelection(new GapCursor($found)));
|
||
return true;
|
||
};
|
||
}
|
||
function handleClick(view, pos, event) {
|
||
if (!view || !view.editable)
|
||
return false;
|
||
let $pos = view.state.doc.resolve(pos);
|
||
if (!GapCursor.valid($pos))
|
||
return false;
|
||
let clickPos = view.posAtCoords({ left: event.clientX, top: event.clientY });
|
||
if (clickPos && clickPos.inside > -1 && NodeSelection.isSelectable(view.state.doc.nodeAt(clickPos.inside)))
|
||
return false;
|
||
view.dispatch(view.state.tr.setSelection(new GapCursor($pos)));
|
||
return true;
|
||
}
|
||
function beforeinput(view, event) {
|
||
if (event.inputType != "insertCompositionText" || !(view.state.selection instanceof GapCursor))
|
||
return false;
|
||
let { $from } = view.state.selection;
|
||
let insert = $from.parent.contentMatchAt($from.index()).findWrapping(view.state.schema.nodes.text);
|
||
if (!insert)
|
||
return false;
|
||
let frag = Fragment.empty;
|
||
for (let i = insert.length - 1; i >= 0; i--)
|
||
frag = Fragment.from(insert[i].createAndFill(null, frag));
|
||
let tr = view.state.tr.replace($from.pos, $from.pos, new Slice(frag, 0, 0));
|
||
tr.setSelection(TextSelection.near(tr.doc.resolve($from.pos + 1)));
|
||
view.dispatch(tr);
|
||
return false;
|
||
}
|
||
function drawGapCursor(state) {
|
||
if (!(state.selection instanceof GapCursor))
|
||
return null;
|
||
let node = document.createElement("div");
|
||
node.className = "ProseMirror-gapcursor";
|
||
return DecorationSet.create(state.doc, [Decoration.widget(state.selection.head, node, { key: "gapcursor" })]);
|
||
}
|
||
|
||
// node_modules/rope-sequence/dist/index.js
|
||
var GOOD_LEAF_SIZE = 200;
|
||
var RopeSequence = function RopeSequence2() {
|
||
};
|
||
RopeSequence.prototype.append = function append(other) {
|
||
if (!other.length) {
|
||
return this;
|
||
}
|
||
other = RopeSequence.from(other);
|
||
return !this.length && other || other.length < GOOD_LEAF_SIZE && this.leafAppend(other) || this.length < GOOD_LEAF_SIZE && other.leafPrepend(this) || this.appendInner(other);
|
||
};
|
||
RopeSequence.prototype.prepend = function prepend(other) {
|
||
if (!other.length) {
|
||
return this;
|
||
}
|
||
return RopeSequence.from(other).append(this);
|
||
};
|
||
RopeSequence.prototype.appendInner = function appendInner(other) {
|
||
return new Append(this, other);
|
||
};
|
||
RopeSequence.prototype.slice = function slice(from2, to) {
|
||
if (from2 === void 0) from2 = 0;
|
||
if (to === void 0) to = this.length;
|
||
if (from2 >= to) {
|
||
return RopeSequence.empty;
|
||
}
|
||
return this.sliceInner(Math.max(0, from2), Math.min(this.length, to));
|
||
};
|
||
RopeSequence.prototype.get = function get(i) {
|
||
if (i < 0 || i >= this.length) {
|
||
return void 0;
|
||
}
|
||
return this.getInner(i);
|
||
};
|
||
RopeSequence.prototype.forEach = function forEach(f, from2, to) {
|
||
if (from2 === void 0) from2 = 0;
|
||
if (to === void 0) to = this.length;
|
||
if (from2 <= to) {
|
||
this.forEachInner(f, from2, to, 0);
|
||
} else {
|
||
this.forEachInvertedInner(f, from2, to, 0);
|
||
}
|
||
};
|
||
RopeSequence.prototype.map = function map(f, from2, to) {
|
||
if (from2 === void 0) from2 = 0;
|
||
if (to === void 0) to = this.length;
|
||
var result = [];
|
||
this.forEach(function(elt, i) {
|
||
return result.push(f(elt, i));
|
||
}, from2, to);
|
||
return result;
|
||
};
|
||
RopeSequence.from = function from(values) {
|
||
if (values instanceof RopeSequence) {
|
||
return values;
|
||
}
|
||
return values && values.length ? new Leaf(values) : RopeSequence.empty;
|
||
};
|
||
var Leaf = function(RopeSequence3) {
|
||
function Leaf2(values) {
|
||
RopeSequence3.call(this);
|
||
this.values = values;
|
||
}
|
||
if (RopeSequence3) Leaf2.__proto__ = RopeSequence3;
|
||
Leaf2.prototype = Object.create(RopeSequence3 && RopeSequence3.prototype);
|
||
Leaf2.prototype.constructor = Leaf2;
|
||
var prototypeAccessors = { length: { configurable: true }, depth: { configurable: true } };
|
||
Leaf2.prototype.flatten = function flatten() {
|
||
return this.values;
|
||
};
|
||
Leaf2.prototype.sliceInner = function sliceInner(from2, to) {
|
||
if (from2 == 0 && to == this.length) {
|
||
return this;
|
||
}
|
||
return new Leaf2(this.values.slice(from2, to));
|
||
};
|
||
Leaf2.prototype.getInner = function getInner(i) {
|
||
return this.values[i];
|
||
};
|
||
Leaf2.prototype.forEachInner = function forEachInner(f, from2, to, start) {
|
||
for (var i = from2; i < to; i++) {
|
||
if (f(this.values[i], start + i) === false) {
|
||
return false;
|
||
}
|
||
}
|
||
};
|
||
Leaf2.prototype.forEachInvertedInner = function forEachInvertedInner(f, from2, to, start) {
|
||
for (var i = from2 - 1; i >= to; i--) {
|
||
if (f(this.values[i], start + i) === false) {
|
||
return false;
|
||
}
|
||
}
|
||
};
|
||
Leaf2.prototype.leafAppend = function leafAppend(other) {
|
||
if (this.length + other.length <= GOOD_LEAF_SIZE) {
|
||
return new Leaf2(this.values.concat(other.flatten()));
|
||
}
|
||
};
|
||
Leaf2.prototype.leafPrepend = function leafPrepend(other) {
|
||
if (this.length + other.length <= GOOD_LEAF_SIZE) {
|
||
return new Leaf2(other.flatten().concat(this.values));
|
||
}
|
||
};
|
||
prototypeAccessors.length.get = function() {
|
||
return this.values.length;
|
||
};
|
||
prototypeAccessors.depth.get = function() {
|
||
return 0;
|
||
};
|
||
Object.defineProperties(Leaf2.prototype, prototypeAccessors);
|
||
return Leaf2;
|
||
}(RopeSequence);
|
||
RopeSequence.empty = new Leaf([]);
|
||
var Append = function(RopeSequence3) {
|
||
function Append2(left, right) {
|
||
RopeSequence3.call(this);
|
||
this.left = left;
|
||
this.right = right;
|
||
this.length = left.length + right.length;
|
||
this.depth = Math.max(left.depth, right.depth) + 1;
|
||
}
|
||
if (RopeSequence3) Append2.__proto__ = RopeSequence3;
|
||
Append2.prototype = Object.create(RopeSequence3 && RopeSequence3.prototype);
|
||
Append2.prototype.constructor = Append2;
|
||
Append2.prototype.flatten = function flatten() {
|
||
return this.left.flatten().concat(this.right.flatten());
|
||
};
|
||
Append2.prototype.getInner = function getInner(i) {
|
||
return i < this.left.length ? this.left.get(i) : this.right.get(i - this.left.length);
|
||
};
|
||
Append2.prototype.forEachInner = function forEachInner(f, from2, to, start) {
|
||
var leftLen = this.left.length;
|
||
if (from2 < leftLen && this.left.forEachInner(f, from2, Math.min(to, leftLen), start) === false) {
|
||
return false;
|
||
}
|
||
if (to > leftLen && this.right.forEachInner(f, Math.max(from2 - leftLen, 0), Math.min(this.length, to) - leftLen, start + leftLen) === false) {
|
||
return false;
|
||
}
|
||
};
|
||
Append2.prototype.forEachInvertedInner = function forEachInvertedInner(f, from2, to, start) {
|
||
var leftLen = this.left.length;
|
||
if (from2 > leftLen && this.right.forEachInvertedInner(f, from2 - leftLen, Math.max(to, leftLen) - leftLen, start + leftLen) === false) {
|
||
return false;
|
||
}
|
||
if (to < leftLen && this.left.forEachInvertedInner(f, Math.min(from2, leftLen), to, start) === false) {
|
||
return false;
|
||
}
|
||
};
|
||
Append2.prototype.sliceInner = function sliceInner(from2, to) {
|
||
if (from2 == 0 && to == this.length) {
|
||
return this;
|
||
}
|
||
var leftLen = this.left.length;
|
||
if (to <= leftLen) {
|
||
return this.left.slice(from2, to);
|
||
}
|
||
if (from2 >= leftLen) {
|
||
return this.right.slice(from2 - leftLen, to - leftLen);
|
||
}
|
||
return this.left.slice(from2, leftLen).append(this.right.slice(0, to - leftLen));
|
||
};
|
||
Append2.prototype.leafAppend = function leafAppend(other) {
|
||
var inner = this.right.leafAppend(other);
|
||
if (inner) {
|
||
return new Append2(this.left, inner);
|
||
}
|
||
};
|
||
Append2.prototype.leafPrepend = function leafPrepend(other) {
|
||
var inner = this.left.leafPrepend(other);
|
||
if (inner) {
|
||
return new Append2(inner, this.right);
|
||
}
|
||
};
|
||
Append2.prototype.appendInner = function appendInner2(other) {
|
||
if (this.left.depth >= Math.max(this.right.depth, other.depth) + 1) {
|
||
return new Append2(this.left, new Append2(this.right, other));
|
||
}
|
||
return new Append2(this, other);
|
||
};
|
||
return Append2;
|
||
}(RopeSequence);
|
||
var dist_default = RopeSequence;
|
||
|
||
// node_modules/prosemirror-history/dist/index.js
|
||
var max_empty_items = 500;
|
||
var Branch = class _Branch {
|
||
constructor(items, eventCount) {
|
||
this.items = items;
|
||
this.eventCount = eventCount;
|
||
}
|
||
// Pop the latest event off the branch's history and apply it
|
||
// to a document transform.
|
||
popEvent(state, preserveItems) {
|
||
if (this.eventCount == 0)
|
||
return null;
|
||
let end = this.items.length;
|
||
for (; ; end--) {
|
||
let next = this.items.get(end - 1);
|
||
if (next.selection) {
|
||
--end;
|
||
break;
|
||
}
|
||
}
|
||
let remap, mapFrom;
|
||
if (preserveItems) {
|
||
remap = this.remapping(end, this.items.length);
|
||
mapFrom = remap.maps.length;
|
||
}
|
||
let transform = state.tr;
|
||
let selection, remaining;
|
||
let addAfter = [], addBefore = [];
|
||
this.items.forEach((item, i) => {
|
||
if (!item.step) {
|
||
if (!remap) {
|
||
remap = this.remapping(end, i + 1);
|
||
mapFrom = remap.maps.length;
|
||
}
|
||
mapFrom--;
|
||
addBefore.push(item);
|
||
return;
|
||
}
|
||
if (remap) {
|
||
addBefore.push(new Item(item.map));
|
||
let step = item.step.map(remap.slice(mapFrom)), map2;
|
||
if (step && transform.maybeStep(step).doc) {
|
||
map2 = transform.mapping.maps[transform.mapping.maps.length - 1];
|
||
addAfter.push(new Item(map2, void 0, void 0, addAfter.length + addBefore.length));
|
||
}
|
||
mapFrom--;
|
||
if (map2)
|
||
remap.appendMap(map2, mapFrom);
|
||
} else {
|
||
transform.maybeStep(item.step);
|
||
}
|
||
if (item.selection) {
|
||
selection = remap ? item.selection.map(remap.slice(mapFrom)) : item.selection;
|
||
remaining = new _Branch(this.items.slice(0, end).append(addBefore.reverse().concat(addAfter)), this.eventCount - 1);
|
||
return false;
|
||
}
|
||
}, this.items.length, 0);
|
||
return { remaining, transform, selection };
|
||
}
|
||
// Create a new branch with the given transform added.
|
||
addTransform(transform, selection, histOptions, preserveItems) {
|
||
let newItems = [], eventCount = this.eventCount;
|
||
let oldItems = this.items, lastItem = !preserveItems && oldItems.length ? oldItems.get(oldItems.length - 1) : null;
|
||
for (let i = 0; i < transform.steps.length; i++) {
|
||
let step = transform.steps[i].invert(transform.docs[i]);
|
||
let item = new Item(transform.mapping.maps[i], step, selection), merged;
|
||
if (merged = lastItem && lastItem.merge(item)) {
|
||
item = merged;
|
||
if (i)
|
||
newItems.pop();
|
||
else
|
||
oldItems = oldItems.slice(0, oldItems.length - 1);
|
||
}
|
||
newItems.push(item);
|
||
if (selection) {
|
||
eventCount++;
|
||
selection = void 0;
|
||
}
|
||
if (!preserveItems)
|
||
lastItem = item;
|
||
}
|
||
let overflow = eventCount - histOptions.depth;
|
||
if (overflow > DEPTH_OVERFLOW) {
|
||
oldItems = cutOffEvents(oldItems, overflow);
|
||
eventCount -= overflow;
|
||
}
|
||
return new _Branch(oldItems.append(newItems), eventCount);
|
||
}
|
||
remapping(from2, to) {
|
||
let maps = new Mapping();
|
||
this.items.forEach((item, i) => {
|
||
let mirrorPos = item.mirrorOffset != null && i - item.mirrorOffset >= from2 ? maps.maps.length - item.mirrorOffset : void 0;
|
||
maps.appendMap(item.map, mirrorPos);
|
||
}, from2, to);
|
||
return maps;
|
||
}
|
||
addMaps(array) {
|
||
if (this.eventCount == 0)
|
||
return this;
|
||
return new _Branch(this.items.append(array.map((map2) => new Item(map2))), this.eventCount);
|
||
}
|
||
// When the collab module receives remote changes, the history has
|
||
// to know about those, so that it can adjust the steps that were
|
||
// rebased on top of the remote changes, and include the position
|
||
// maps for the remote changes in its array of items.
|
||
rebased(rebasedTransform, rebasedCount) {
|
||
if (!this.eventCount)
|
||
return this;
|
||
let rebasedItems = [], start = Math.max(0, this.items.length - rebasedCount);
|
||
let mapping = rebasedTransform.mapping;
|
||
let newUntil = rebasedTransform.steps.length;
|
||
let eventCount = this.eventCount;
|
||
this.items.forEach((item) => {
|
||
if (item.selection)
|
||
eventCount--;
|
||
}, start);
|
||
let iRebased = rebasedCount;
|
||
this.items.forEach((item) => {
|
||
let pos = mapping.getMirror(--iRebased);
|
||
if (pos == null)
|
||
return;
|
||
newUntil = Math.min(newUntil, pos);
|
||
let map2 = mapping.maps[pos];
|
||
if (item.step) {
|
||
let step = rebasedTransform.steps[pos].invert(rebasedTransform.docs[pos]);
|
||
let selection = item.selection && item.selection.map(mapping.slice(iRebased + 1, pos));
|
||
if (selection)
|
||
eventCount++;
|
||
rebasedItems.push(new Item(map2, step, selection));
|
||
} else {
|
||
rebasedItems.push(new Item(map2));
|
||
}
|
||
}, start);
|
||
let newMaps = [];
|
||
for (let i = rebasedCount; i < newUntil; i++)
|
||
newMaps.push(new Item(mapping.maps[i]));
|
||
let items = this.items.slice(0, start).append(newMaps).append(rebasedItems);
|
||
let branch = new _Branch(items, eventCount);
|
||
if (branch.emptyItemCount() > max_empty_items)
|
||
branch = branch.compress(this.items.length - rebasedItems.length);
|
||
return branch;
|
||
}
|
||
emptyItemCount() {
|
||
let count = 0;
|
||
this.items.forEach((item) => {
|
||
if (!item.step)
|
||
count++;
|
||
});
|
||
return count;
|
||
}
|
||
// Compressing a branch means rewriting it to push the air (map-only
|
||
// items) out. During collaboration, these naturally accumulate
|
||
// because each remote change adds one. The `upto` argument is used
|
||
// to ensure that only the items below a given level are compressed,
|
||
// because `rebased` relies on a clean, untouched set of items in
|
||
// order to associate old items with rebased steps.
|
||
compress(upto = this.items.length) {
|
||
let remap = this.remapping(0, upto), mapFrom = remap.maps.length;
|
||
let items = [], events = 0;
|
||
this.items.forEach((item, i) => {
|
||
if (i >= upto) {
|
||
items.push(item);
|
||
if (item.selection)
|
||
events++;
|
||
} else if (item.step) {
|
||
let step = item.step.map(remap.slice(mapFrom)), map2 = step && step.getMap();
|
||
mapFrom--;
|
||
if (map2)
|
||
remap.appendMap(map2, mapFrom);
|
||
if (step) {
|
||
let selection = item.selection && item.selection.map(remap.slice(mapFrom));
|
||
if (selection)
|
||
events++;
|
||
let newItem = new Item(map2.invert(), step, selection), merged, last = items.length - 1;
|
||
if (merged = items.length && items[last].merge(newItem))
|
||
items[last] = merged;
|
||
else
|
||
items.push(newItem);
|
||
}
|
||
} else if (item.map) {
|
||
mapFrom--;
|
||
}
|
||
}, this.items.length, 0);
|
||
return new _Branch(dist_default.from(items.reverse()), events);
|
||
}
|
||
};
|
||
Branch.empty = new Branch(dist_default.empty, 0);
|
||
function cutOffEvents(items, n) {
|
||
let cutPoint;
|
||
items.forEach((item, i) => {
|
||
if (item.selection && n-- == 0) {
|
||
cutPoint = i;
|
||
return false;
|
||
}
|
||
});
|
||
return items.slice(cutPoint);
|
||
}
|
||
var Item = class _Item {
|
||
constructor(map2, step, selection, mirrorOffset) {
|
||
this.map = map2;
|
||
this.step = step;
|
||
this.selection = selection;
|
||
this.mirrorOffset = mirrorOffset;
|
||
}
|
||
merge(other) {
|
||
if (this.step && other.step && !other.selection) {
|
||
let step = other.step.merge(this.step);
|
||
if (step)
|
||
return new _Item(step.getMap().invert(), step, this.selection);
|
||
}
|
||
}
|
||
};
|
||
var HistoryState = class {
|
||
constructor(done, undone, prevRanges, prevTime, prevComposition) {
|
||
this.done = done;
|
||
this.undone = undone;
|
||
this.prevRanges = prevRanges;
|
||
this.prevTime = prevTime;
|
||
this.prevComposition = prevComposition;
|
||
}
|
||
};
|
||
var DEPTH_OVERFLOW = 20;
|
||
function applyTransaction(history2, state, tr, options) {
|
||
let historyTr = tr.getMeta(historyKey), rebased;
|
||
if (historyTr)
|
||
return historyTr.historyState;
|
||
if (tr.getMeta(closeHistoryKey))
|
||
history2 = new HistoryState(history2.done, history2.undone, null, 0, -1);
|
||
let appended = tr.getMeta("appendedTransaction");
|
||
if (tr.steps.length == 0) {
|
||
return history2;
|
||
} else if (appended && appended.getMeta(historyKey)) {
|
||
if (appended.getMeta(historyKey).redo)
|
||
return new HistoryState(history2.done.addTransform(tr, void 0, options, mustPreserveItems(state)), history2.undone, rangesFor(tr.mapping.maps), history2.prevTime, history2.prevComposition);
|
||
else
|
||
return new HistoryState(history2.done, history2.undone.addTransform(tr, void 0, options, mustPreserveItems(state)), null, history2.prevTime, history2.prevComposition);
|
||
} else if (tr.getMeta("addToHistory") !== false && !(appended && appended.getMeta("addToHistory") === false)) {
|
||
let composition = tr.getMeta("composition");
|
||
let newGroup = history2.prevTime == 0 || !appended && history2.prevComposition != composition && (history2.prevTime < (tr.time || 0) - options.newGroupDelay || !isAdjacentTo(tr, history2.prevRanges));
|
||
let prevRanges = appended ? mapRanges(history2.prevRanges, tr.mapping) : rangesFor(tr.mapping.maps);
|
||
return new HistoryState(history2.done.addTransform(tr, newGroup ? state.selection.getBookmark() : void 0, options, mustPreserveItems(state)), Branch.empty, prevRanges, tr.time, composition == null ? history2.prevComposition : composition);
|
||
} else if (rebased = tr.getMeta("rebased")) {
|
||
return new HistoryState(history2.done.rebased(tr, rebased), history2.undone.rebased(tr, rebased), mapRanges(history2.prevRanges, tr.mapping), history2.prevTime, history2.prevComposition);
|
||
} else {
|
||
return new HistoryState(history2.done.addMaps(tr.mapping.maps), history2.undone.addMaps(tr.mapping.maps), mapRanges(history2.prevRanges, tr.mapping), history2.prevTime, history2.prevComposition);
|
||
}
|
||
}
|
||
function isAdjacentTo(transform, prevRanges) {
|
||
if (!prevRanges)
|
||
return false;
|
||
if (!transform.docChanged)
|
||
return true;
|
||
let adjacent = false;
|
||
transform.mapping.maps[0].forEach((start, end) => {
|
||
for (let i = 0; i < prevRanges.length; i += 2)
|
||
if (start <= prevRanges[i + 1] && end >= prevRanges[i])
|
||
adjacent = true;
|
||
});
|
||
return adjacent;
|
||
}
|
||
function rangesFor(maps) {
|
||
let result = [];
|
||
for (let i = maps.length - 1; i >= 0 && result.length == 0; i--)
|
||
maps[i].forEach((_from, _to, from2, to) => result.push(from2, to));
|
||
return result;
|
||
}
|
||
function mapRanges(ranges, mapping) {
|
||
if (!ranges)
|
||
return null;
|
||
let result = [];
|
||
for (let i = 0; i < ranges.length; i += 2) {
|
||
let from2 = mapping.map(ranges[i], 1), to = mapping.map(ranges[i + 1], -1);
|
||
if (from2 <= to)
|
||
result.push(from2, to);
|
||
}
|
||
return result;
|
||
}
|
||
function histTransaction(history2, state, redo2) {
|
||
let preserveItems = mustPreserveItems(state);
|
||
let histOptions = historyKey.get(state).spec.config;
|
||
let pop = (redo2 ? history2.undone : history2.done).popEvent(state, preserveItems);
|
||
if (!pop)
|
||
return null;
|
||
let selection = pop.selection.resolve(pop.transform.doc);
|
||
let added = (redo2 ? history2.done : history2.undone).addTransform(pop.transform, state.selection.getBookmark(), histOptions, preserveItems);
|
||
let newHist = new HistoryState(redo2 ? added : pop.remaining, redo2 ? pop.remaining : added, null, 0, -1);
|
||
return pop.transform.setSelection(selection).setMeta(historyKey, { redo: redo2, historyState: newHist });
|
||
}
|
||
var cachedPreserveItems = false;
|
||
var cachedPreserveItemsPlugins = null;
|
||
function mustPreserveItems(state) {
|
||
let plugins = state.plugins;
|
||
if (cachedPreserveItemsPlugins != plugins) {
|
||
cachedPreserveItems = false;
|
||
cachedPreserveItemsPlugins = plugins;
|
||
for (let i = 0; i < plugins.length; i++)
|
||
if (plugins[i].spec.historyPreserveItems) {
|
||
cachedPreserveItems = true;
|
||
break;
|
||
}
|
||
}
|
||
return cachedPreserveItems;
|
||
}
|
||
var historyKey = new PluginKey("history");
|
||
var closeHistoryKey = new PluginKey("closeHistory");
|
||
function history(config = {}) {
|
||
config = {
|
||
depth: config.depth || 100,
|
||
newGroupDelay: config.newGroupDelay || 500
|
||
};
|
||
return new Plugin({
|
||
key: historyKey,
|
||
state: {
|
||
init() {
|
||
return new HistoryState(Branch.empty, Branch.empty, null, 0, -1);
|
||
},
|
||
apply(tr, hist, state) {
|
||
return applyTransaction(hist, state, tr, config);
|
||
}
|
||
},
|
||
config,
|
||
props: {
|
||
handleDOMEvents: {
|
||
beforeinput(view, e) {
|
||
let inputType = e.inputType;
|
||
let command = inputType == "historyUndo" ? undo : inputType == "historyRedo" ? redo : null;
|
||
if (!command || !view.editable)
|
||
return false;
|
||
e.preventDefault();
|
||
return command(view.state, view.dispatch);
|
||
}
|
||
}
|
||
}
|
||
});
|
||
}
|
||
function buildCommand(redo2, scroll) {
|
||
return (state, dispatch) => {
|
||
let hist = historyKey.getState(state);
|
||
if (!hist || (redo2 ? hist.undone : hist.done).eventCount == 0)
|
||
return false;
|
||
if (dispatch) {
|
||
let tr = histTransaction(hist, state, redo2);
|
||
if (tr)
|
||
dispatch(scroll ? tr.scrollIntoView() : tr);
|
||
}
|
||
return true;
|
||
};
|
||
}
|
||
var undo = buildCommand(false, true);
|
||
var redo = buildCommand(true, true);
|
||
var undoNoScroll = buildCommand(false, false);
|
||
var redoNoScroll = buildCommand(true, false);
|
||
|
||
// node_modules/@tiptap/extensions/dist/index.js
|
||
var CharacterCount = Extension.create({
|
||
name: "characterCount",
|
||
addOptions() {
|
||
return {
|
||
limit: null,
|
||
mode: "textSize",
|
||
textCounter: (text) => text.length,
|
||
wordCounter: (text) => text.split(" ").filter((word) => word !== "").length
|
||
};
|
||
},
|
||
addStorage() {
|
||
return {
|
||
characters: () => 0,
|
||
words: () => 0
|
||
};
|
||
},
|
||
onBeforeCreate() {
|
||
this.storage.characters = (options) => {
|
||
const node = (options == null ? void 0 : options.node) || this.editor.state.doc;
|
||
const mode = (options == null ? void 0 : options.mode) || this.options.mode;
|
||
if (mode === "textSize") {
|
||
const text = node.textBetween(0, node.content.size, void 0, " ");
|
||
return this.options.textCounter(text);
|
||
}
|
||
return node.nodeSize;
|
||
};
|
||
this.storage.words = (options) => {
|
||
const node = (options == null ? void 0 : options.node) || this.editor.state.doc;
|
||
const text = node.textBetween(0, node.content.size, " ", " ");
|
||
return this.options.wordCounter(text);
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
let initialEvaluationDone = false;
|
||
return [
|
||
new Plugin({
|
||
key: new PluginKey("characterCount"),
|
||
appendTransaction: (transactions, oldState, newState) => {
|
||
if (initialEvaluationDone) {
|
||
return;
|
||
}
|
||
const limit = this.options.limit;
|
||
if (limit === null || limit === void 0 || limit === 0) {
|
||
initialEvaluationDone = true;
|
||
return;
|
||
}
|
||
const initialContentSize = this.storage.characters({ node: newState.doc });
|
||
if (initialContentSize > limit) {
|
||
const over = initialContentSize - limit;
|
||
const from2 = 0;
|
||
const to = over;
|
||
console.warn(
|
||
`[CharacterCount] Initial content exceeded limit of ${limit} characters. Content was automatically trimmed.`
|
||
);
|
||
const tr = newState.tr.deleteRange(from2, to);
|
||
initialEvaluationDone = true;
|
||
return tr;
|
||
}
|
||
initialEvaluationDone = true;
|
||
},
|
||
filterTransaction: (transaction, state) => {
|
||
const limit = this.options.limit;
|
||
if (!transaction.docChanged || limit === 0 || limit === null || limit === void 0) {
|
||
return true;
|
||
}
|
||
const oldSize = this.storage.characters({ node: state.doc });
|
||
const newSize = this.storage.characters({ node: transaction.doc });
|
||
if (newSize <= limit) {
|
||
return true;
|
||
}
|
||
if (oldSize > limit && newSize > limit && newSize <= oldSize) {
|
||
return true;
|
||
}
|
||
if (oldSize > limit && newSize > limit && newSize > oldSize) {
|
||
return false;
|
||
}
|
||
const isPaste = transaction.getMeta("paste");
|
||
if (!isPaste) {
|
||
return false;
|
||
}
|
||
const pos = transaction.selection.$head.pos;
|
||
const over = newSize - limit;
|
||
const from2 = pos - over;
|
||
const to = pos;
|
||
transaction.deleteRange(from2, to);
|
||
const updatedSize = this.storage.characters({ node: transaction.doc });
|
||
if (updatedSize > limit) {
|
||
return false;
|
||
}
|
||
return true;
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
var Dropcursor = Extension.create({
|
||
name: "dropCursor",
|
||
addOptions() {
|
||
return {
|
||
color: "currentColor",
|
||
width: 1,
|
||
class: void 0
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
return [dropCursor(this.options)];
|
||
}
|
||
});
|
||
var Focus = Extension.create({
|
||
name: "focus",
|
||
addOptions() {
|
||
return {
|
||
className: "has-focus",
|
||
mode: "all"
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
return [
|
||
new Plugin({
|
||
key: new PluginKey("focus"),
|
||
props: {
|
||
decorations: ({ doc, selection }) => {
|
||
const { isEditable, isFocused } = this.editor;
|
||
const { anchor } = selection;
|
||
const decorations = [];
|
||
if (!isEditable || !isFocused) {
|
||
return DecorationSet.create(doc, []);
|
||
}
|
||
let maxLevels = 0;
|
||
if (this.options.mode === "deepest") {
|
||
doc.descendants((node, pos) => {
|
||
if (node.isText) {
|
||
return;
|
||
}
|
||
const isCurrent = anchor >= pos && anchor <= pos + node.nodeSize - 1;
|
||
if (!isCurrent) {
|
||
return false;
|
||
}
|
||
maxLevels += 1;
|
||
});
|
||
}
|
||
let currentLevel = 0;
|
||
doc.descendants((node, pos) => {
|
||
if (node.isText) {
|
||
return false;
|
||
}
|
||
const isCurrent = anchor >= pos && anchor <= pos + node.nodeSize - 1;
|
||
if (!isCurrent) {
|
||
return false;
|
||
}
|
||
currentLevel += 1;
|
||
const outOfScope = this.options.mode === "deepest" && maxLevels - currentLevel > 0 || this.options.mode === "shallowest" && currentLevel > 1;
|
||
if (outOfScope) {
|
||
return this.options.mode === "deepest";
|
||
}
|
||
decorations.push(
|
||
Decoration.node(pos, pos + node.nodeSize, {
|
||
class: this.options.className
|
||
})
|
||
);
|
||
});
|
||
return DecorationSet.create(doc, decorations);
|
||
}
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
var Gapcursor = Extension.create({
|
||
name: "gapCursor",
|
||
addProseMirrorPlugins() {
|
||
return [gapCursor()];
|
||
},
|
||
extendNodeSchema(extension) {
|
||
var _a;
|
||
const context = {
|
||
name: extension.name,
|
||
options: extension.options,
|
||
storage: extension.storage
|
||
};
|
||
return {
|
||
allowGapCursor: (_a = callOrReturn(getExtensionField(extension, "allowGapCursor", context))) != null ? _a : null
|
||
};
|
||
}
|
||
});
|
||
var DEFAULT_DATA_ATTRIBUTE = "placeholder";
|
||
function preparePlaceholderAttribute(attr) {
|
||
return attr.replace(/\s+/g, "-").replace(/[^a-zA-Z0-9-]/g, "").replace(/^[0-9-]+/, "").replace(/^-+/, "").toLowerCase();
|
||
}
|
||
var Placeholder = Extension.create({
|
||
name: "placeholder",
|
||
addOptions() {
|
||
return {
|
||
emptyEditorClass: "is-editor-empty",
|
||
emptyNodeClass: "is-empty",
|
||
dataAttribute: DEFAULT_DATA_ATTRIBUTE,
|
||
placeholder: "Write something …",
|
||
showOnlyWhenEditable: true,
|
||
showOnlyCurrent: true,
|
||
includeChildren: false
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
const dataAttribute = this.options.dataAttribute ? `data-${preparePlaceholderAttribute(this.options.dataAttribute)}` : `data-${DEFAULT_DATA_ATTRIBUTE}`;
|
||
return [
|
||
new Plugin({
|
||
key: new PluginKey("placeholder"),
|
||
props: {
|
||
decorations: ({ doc, selection }) => {
|
||
const active = this.editor.isEditable || !this.options.showOnlyWhenEditable;
|
||
const { anchor } = selection;
|
||
const decorations = [];
|
||
if (!active) {
|
||
return null;
|
||
}
|
||
const isEmptyDoc = this.editor.isEmpty;
|
||
doc.descendants((node, pos) => {
|
||
const hasAnchor = anchor >= pos && anchor <= pos + node.nodeSize;
|
||
const isEmpty = !node.isLeaf && isNodeEmpty(node);
|
||
if ((hasAnchor || !this.options.showOnlyCurrent) && isEmpty) {
|
||
const classes = [this.options.emptyNodeClass];
|
||
if (isEmptyDoc) {
|
||
classes.push(this.options.emptyEditorClass);
|
||
}
|
||
const decoration = Decoration.node(pos, pos + node.nodeSize, {
|
||
class: classes.join(" "),
|
||
[dataAttribute]: typeof this.options.placeholder === "function" ? this.options.placeholder({
|
||
editor: this.editor,
|
||
node,
|
||
pos,
|
||
hasAnchor
|
||
}) : this.options.placeholder
|
||
});
|
||
decorations.push(decoration);
|
||
}
|
||
return this.options.includeChildren;
|
||
});
|
||
return DecorationSet.create(doc, decorations);
|
||
}
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
var Selection2 = Extension.create({
|
||
name: "selection",
|
||
addOptions() {
|
||
return {
|
||
className: "selection"
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
const { editor, options } = this;
|
||
return [
|
||
new Plugin({
|
||
key: new PluginKey("selection"),
|
||
props: {
|
||
decorations(state) {
|
||
if (state.selection.empty || editor.isFocused || !editor.isEditable || isNodeSelection(state.selection) || editor.view.dragging) {
|
||
return null;
|
||
}
|
||
return DecorationSet.create(state.doc, [
|
||
Decoration.inline(state.selection.from, state.selection.to, {
|
||
class: options.className
|
||
})
|
||
]);
|
||
}
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
function nodeEqualsType({ types, node }) {
|
||
return node && Array.isArray(types) && types.includes(node.type) || (node == null ? void 0 : node.type) === types;
|
||
}
|
||
var TrailingNode = Extension.create({
|
||
name: "trailingNode",
|
||
addOptions() {
|
||
return {
|
||
node: void 0,
|
||
notAfter: []
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
var _a;
|
||
const plugin = new PluginKey(this.name);
|
||
const defaultNode = this.options.node || ((_a = this.editor.schema.topNodeType.contentMatch.defaultType) == null ? void 0 : _a.name) || "paragraph";
|
||
const disabledNodes = Object.entries(this.editor.schema.nodes).map(([, value]) => value).filter((node) => (this.options.notAfter || []).concat(defaultNode).includes(node.name));
|
||
return [
|
||
new Plugin({
|
||
key: plugin,
|
||
appendTransaction: (_, __, state) => {
|
||
const { doc, tr, schema } = state;
|
||
const shouldInsertNodeAtEnd = plugin.getState(state);
|
||
const endPosition = doc.content.size;
|
||
const type = schema.nodes[defaultNode];
|
||
if (!shouldInsertNodeAtEnd) {
|
||
return;
|
||
}
|
||
return tr.insert(endPosition, type.create());
|
||
},
|
||
state: {
|
||
init: (_, state) => {
|
||
const lastNode = state.tr.doc.lastChild;
|
||
return !nodeEqualsType({ node: lastNode, types: disabledNodes });
|
||
},
|
||
apply: (tr, value) => {
|
||
if (!tr.docChanged) {
|
||
return value;
|
||
}
|
||
if (tr.getMeta("__uniqueIDTransaction")) {
|
||
return value;
|
||
}
|
||
const lastNode = tr.doc.lastChild;
|
||
return !nodeEqualsType({ node: lastNode, types: disabledNodes });
|
||
}
|
||
}
|
||
})
|
||
];
|
||
}
|
||
});
|
||
var UndoRedo = Extension.create({
|
||
name: "undoRedo",
|
||
addOptions() {
|
||
return {
|
||
depth: 100,
|
||
newGroupDelay: 500
|
||
};
|
||
},
|
||
addCommands() {
|
||
return {
|
||
undo: () => ({ state, dispatch }) => {
|
||
return undo(state, dispatch);
|
||
},
|
||
redo: () => ({ state, dispatch }) => {
|
||
return redo(state, dispatch);
|
||
}
|
||
};
|
||
},
|
||
addProseMirrorPlugins() {
|
||
return [history(this.options)];
|
||
},
|
||
addKeyboardShortcuts() {
|
||
return {
|
||
"Mod-z": () => this.editor.commands.undo(),
|
||
"Shift-Mod-z": () => this.editor.commands.redo(),
|
||
"Mod-y": () => this.editor.commands.redo(),
|
||
// Russian keyboard layouts
|
||
"Mod-я": () => this.editor.commands.undo(),
|
||
"Shift-Mod-я": () => this.editor.commands.redo()
|
||
};
|
||
}
|
||
});
|
||
|
||
// node_modules/@tiptap/starter-kit/dist/index.js
|
||
var StarterKit = Extension.create({
|
||
name: "starterKit",
|
||
addExtensions() {
|
||
var _a, _b, _c, _d;
|
||
const extensions = [];
|
||
if (this.options.bold !== false) {
|
||
extensions.push(Bold.configure(this.options.bold));
|
||
}
|
||
if (this.options.blockquote !== false) {
|
||
extensions.push(Blockquote.configure(this.options.blockquote));
|
||
}
|
||
if (this.options.bulletList !== false) {
|
||
extensions.push(BulletList.configure(this.options.bulletList));
|
||
}
|
||
if (this.options.code !== false) {
|
||
extensions.push(Code.configure(this.options.code));
|
||
}
|
||
if (this.options.codeBlock !== false) {
|
||
extensions.push(CodeBlock.configure(this.options.codeBlock));
|
||
}
|
||
if (this.options.document !== false) {
|
||
extensions.push(Document.configure(this.options.document));
|
||
}
|
||
if (this.options.dropcursor !== false) {
|
||
extensions.push(Dropcursor.configure(this.options.dropcursor));
|
||
}
|
||
if (this.options.gapcursor !== false) {
|
||
extensions.push(Gapcursor.configure(this.options.gapcursor));
|
||
}
|
||
if (this.options.hardBreak !== false) {
|
||
extensions.push(HardBreak.configure(this.options.hardBreak));
|
||
}
|
||
if (this.options.heading !== false) {
|
||
extensions.push(Heading.configure(this.options.heading));
|
||
}
|
||
if (this.options.undoRedo !== false) {
|
||
extensions.push(UndoRedo.configure(this.options.undoRedo));
|
||
}
|
||
if (this.options.horizontalRule !== false) {
|
||
extensions.push(HorizontalRule.configure(this.options.horizontalRule));
|
||
}
|
||
if (this.options.italic !== false) {
|
||
extensions.push(Italic.configure(this.options.italic));
|
||
}
|
||
if (this.options.listItem !== false) {
|
||
extensions.push(ListItem.configure(this.options.listItem));
|
||
}
|
||
if (this.options.listKeymap !== false) {
|
||
extensions.push(ListKeymap.configure((_a = this.options) == null ? void 0 : _a.listKeymap));
|
||
}
|
||
if (this.options.link !== false) {
|
||
extensions.push(Link.configure((_b = this.options) == null ? void 0 : _b.link));
|
||
}
|
||
if (this.options.orderedList !== false) {
|
||
extensions.push(OrderedList.configure(this.options.orderedList));
|
||
}
|
||
if (this.options.paragraph !== false) {
|
||
extensions.push(Paragraph.configure(this.options.paragraph));
|
||
}
|
||
if (this.options.strike !== false) {
|
||
extensions.push(Strike.configure(this.options.strike));
|
||
}
|
||
if (this.options.text !== false) {
|
||
extensions.push(Text.configure(this.options.text));
|
||
}
|
||
if (this.options.underline !== false) {
|
||
extensions.push(Underline.configure((_c = this.options) == null ? void 0 : _c.underline));
|
||
}
|
||
if (this.options.trailingNode !== false) {
|
||
extensions.push(TrailingNode.configure((_d = this.options) == null ? void 0 : _d.trailingNode));
|
||
}
|
||
return extensions;
|
||
}
|
||
});
|
||
var index_default = StarterKit;
|
||
export {
|
||
StarterKit,
|
||
index_default as default
|
||
};
|
||
//# sourceMappingURL=@tiptap_starter-kit.js.map
|