mirror of
https://github.com/naturalcrit/homebrewery.git
synced 2026-03-25 19:18:11 +00:00
229 lines
6.0 KiB
JavaScript
229 lines
6.0 KiB
JavaScript
// customMarkdownGrammar.js
|
|
|
|
// --- Custom tags with CM6-compatible class names ---
|
|
export const customTags = {
|
|
pageLine: "pageLine", // .cm-pageLine
|
|
snippetLine: "snippetLine", // .cm-snippetLine
|
|
columnSplit: "columnSplit", // .cm-columnSplit
|
|
snippetBreak: "snippetBreak", // .cm-snippetBreak
|
|
block: "block", // .cm-block
|
|
inlineBlock: "inline-block", // .cm-inline-block
|
|
injection: "injection", // .cm-injection
|
|
emoji: "emoji", // .cm-emoji
|
|
superscript: "superscript", // .cm-superscript
|
|
subscript: "subscript", // .cm-subscript
|
|
definitionList: "definitionList", // .cm-definitionList
|
|
definitionTerm: "definitionTerm", // .cm-definitionTerm
|
|
definitionDesc: "definitionDesc", // .cm-definitionDesc
|
|
definitionColon: "definitionColon", // .cm-definitionColon
|
|
};
|
|
|
|
// --- Tokenizer function ---
|
|
export function tokenizeCustomMarkdown(text) {
|
|
const tokens = [];
|
|
const lines = text.split("\n");
|
|
|
|
// Track multi-line blocks
|
|
let inBlock = false;
|
|
let blockStart = 0;
|
|
|
|
lines.forEach((lineText, lineNumber) => {
|
|
// --- Page / snippet lines ---
|
|
if (/\\page/.test(lineText)) tokens.push({ line: lineNumber, type: customTags.pageLine });
|
|
if (/\\snippet/.test(lineText)) tokens.push({ line: lineNumber, type: customTags.snippetLine });
|
|
if (/^\\column(?:break)?$/.test(lineText)) tokens.push({ line: lineNumber, type: customTags.columnSplit });
|
|
if (/\\snippet/.test(lineText)) tokens.push({ line: lineNumber, type: customTags.snippetBreak });
|
|
|
|
// --- Emoji ---
|
|
if (/:\w+?:/.test(lineText)) tokens.push({ line: lineNumber, type: customTags.emoji });
|
|
|
|
// --- Superscript / Subscript ---
|
|
if (/\^/.test(lineText)) {
|
|
let startIndex = lineText.indexOf("^");
|
|
const superRegex = /\^(?!\s)(?=([^\n\^]*[^\s\^]))\1\^/gy;
|
|
const subRegex = /\^\^(?!\s)(?=([^\n\^]*[^\s\^]))\1\^\^/gy;
|
|
|
|
while (startIndex >= 0) {
|
|
superRegex.lastIndex = subRegex.lastIndex = startIndex;
|
|
|
|
let match = subRegex.exec(lineText);
|
|
let type = customTags.subscript;
|
|
|
|
if (!match) {
|
|
match = superRegex.exec(lineText);
|
|
type = customTags.superscript;
|
|
}
|
|
|
|
if (match) {
|
|
tokens.push({
|
|
line: lineNumber,
|
|
type,
|
|
from: match.index,
|
|
to: match.index + match[0].length,
|
|
});
|
|
}
|
|
|
|
startIndex = lineText.indexOf(
|
|
"^",
|
|
Math.max(startIndex + 1, superRegex.lastIndex || 0, subRegex.lastIndex || 0),
|
|
);
|
|
}
|
|
}
|
|
|
|
// --- inline definition lists ---
|
|
if (/::/.test(lineText)) {
|
|
if (/^:*$/.test(lineText) == true) {
|
|
return; //if line only has colons, stops
|
|
}
|
|
|
|
const singleLineRegex = /^([^:\n]*\S)(::)([^\n]*)$/dmy;
|
|
|
|
let match = singleLineRegex.exec(lineText);
|
|
|
|
if (match) {
|
|
const [full, term, colons, desc] = match;
|
|
let offset = 0;
|
|
|
|
// Entire line as definitionList
|
|
tokens.push({
|
|
line: lineNumber,
|
|
type: customTags.definitionList,
|
|
});
|
|
|
|
// Term
|
|
tokens.push({
|
|
line: lineNumber,
|
|
type: customTags.definitionTerm,
|
|
from: offset,
|
|
to: offset + term.length,
|
|
});
|
|
offset += term.length;
|
|
|
|
// ::
|
|
tokens.push({
|
|
line: lineNumber,
|
|
type: customTags.definitionColon,
|
|
from: offset,
|
|
to: offset + colons.length,
|
|
});
|
|
offset += colons.length;
|
|
|
|
// Definition
|
|
tokens.push({
|
|
line: lineNumber,
|
|
type: customTags.definitionDesc,
|
|
from: offset,
|
|
to: offset + desc.length,
|
|
});
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
// multiline def list
|
|
if (!/^::/.test(lines[lineNumber]) && lineNumber + 1 < lines.length && /^::/.test(lines[lineNumber + 1])) {
|
|
const term = lineText;
|
|
const startLine = lineNumber;
|
|
let defs = [];
|
|
let endLine = startLine;
|
|
|
|
// collect all following :: definitions
|
|
for (let i = lineNumber + 1; i < lines.length; i++) {
|
|
const nextLine = lines[i];
|
|
const onlyColonsMatch = /^:*$/.test(nextLine);
|
|
const defMatch = /^(::)(.*\S.*)?\s*$/.exec(nextLine);
|
|
if (!onlyColonsMatch && defMatch) {
|
|
defs.push({ colons: defMatch[1], desc: defMatch[2], line: i });
|
|
endLine = i;
|
|
} else break;
|
|
}
|
|
|
|
if (defs.length > 0) {
|
|
tokens.push({
|
|
line: startLine,
|
|
type: customTags.definitionList,
|
|
});
|
|
|
|
// term
|
|
tokens.push({
|
|
line: startLine,
|
|
type: customTags.definitionTerm,
|
|
from: 0,
|
|
to: lineText.length,
|
|
});
|
|
|
|
// definitions
|
|
defs.forEach((d) => {
|
|
tokens.push({
|
|
line: d.line,
|
|
type: customTags.definitionList,
|
|
});
|
|
|
|
tokens.push({
|
|
line: d.line,
|
|
type: customTags.definitionColon,
|
|
from: 0,
|
|
to: d.colons.length,
|
|
});
|
|
tokens.push({
|
|
line: d.line,
|
|
type: customTags.definitionDesc,
|
|
from: d.colons.length,
|
|
to: d.colons.length + d.desc.length,
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
if (lineText.includes("{") && lineText.includes("}")) {
|
|
const injectionRegex = /(?:^|[^{\n])({(?=((?:[:=](?:"[\w,\-()#%. ]*"|[\w\-()#%.]*)|[^"':={}\s]*)*))\2})/gm;
|
|
let match;
|
|
while ((match = injectionRegex.exec(lineText)) !== null) {
|
|
tokens.push({
|
|
line: lineNumber,
|
|
from: match.index +1,
|
|
to: match.index + match[1].length +1,
|
|
type: customTags.injection,
|
|
});
|
|
console.log(match);
|
|
}
|
|
}
|
|
if (lineText.includes("{{") && lineText.includes("}}")) {
|
|
// Inline blocks: single-line {{…}}
|
|
const spanRegex = /{{(?=((?:[:=](?:"[\w,\-()#%. ]*"|[\w\-()#%.]*)|[^"':={}\s]*)*))\1 *|}}/g;
|
|
let match;
|
|
let blockCount = 0;
|
|
while ((match = spanRegex.exec(lineText)) !== null) {
|
|
if (match[0].startsWith("{{")) {
|
|
blockCount += 1;
|
|
} else {
|
|
blockCount -= 1;
|
|
}
|
|
if (blockCount < 0) {
|
|
blockCount = 0;
|
|
continue;
|
|
}
|
|
tokens.push({
|
|
line: lineNumber,
|
|
from: match.index,
|
|
to: match.index + match[0].length,
|
|
type: customTags.inlineBlock,
|
|
});
|
|
}
|
|
} else if (lineText.trimLeft().startsWith("{{") || lineText.trimLeft().startsWith("}}")) {
|
|
// Highlight block divs {{\n Content \n}}
|
|
let endCh = lineText.length + 1;
|
|
|
|
const match = lineText.match(
|
|
/^ *{{(?=((?:[:=](?:"[\w,\-()#%. ]*"|[\w\-()#%.]*)|[^"':={}\s]*)*))\1 *$|^ *}}$/,
|
|
);
|
|
if (match) endCh = match.index + match[0].length;
|
|
tokens.push({ line: lineNumber, type: customTags.block });
|
|
}
|
|
|
|
|
|
});
|
|
|
|
return tokens;
|
|
}
|