2023-09-01 22:57:29 +08:00
|
|
|
import {
|
|
|
|
BlockContext,
|
|
|
|
LeafBlock,
|
|
|
|
LeafBlockParser,
|
|
|
|
MarkdownConfig,
|
|
|
|
} from "../deps.ts";
|
|
|
|
|
|
|
|
import { tags as t } from "@lezer/highlight";
|
|
|
|
import { TaskStateTag } from "./customtags.ts";
|
|
|
|
|
|
|
|
// Taken from https://github.com/lezer-parser/markdown/blob/main/src/extension.ts and adapted
|
|
|
|
|
|
|
|
class MultiStatusTaskParser implements LeafBlockParser {
|
|
|
|
constructor(private status: string) {
|
|
|
|
}
|
|
|
|
|
|
|
|
nextLine() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
finish(cx: BlockContext, leaf: LeafBlock) {
|
|
|
|
cx.addLeafElement(
|
|
|
|
leaf,
|
|
|
|
cx.elt("Task", leaf.start, leaf.start + leaf.content.length, [
|
|
|
|
cx.elt("TaskState", leaf.start, leaf.start + 2 + this.status.length, [
|
2023-10-03 20:16:33 +08:00
|
|
|
cx.elt("TaskMark", leaf.start, leaf.start + 1),
|
2023-09-01 22:57:29 +08:00
|
|
|
cx.elt(
|
2023-10-03 20:16:33 +08:00
|
|
|
"TaskMark",
|
2023-09-01 22:57:29 +08:00
|
|
|
leaf.start + 1 + this.status.length,
|
|
|
|
leaf.start + 2 + this.status.length,
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
...cx.parser.parseInline(
|
|
|
|
leaf.content.slice(this.status.length + 2),
|
|
|
|
leaf.start + this.status.length + 2,
|
|
|
|
),
|
|
|
|
]),
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const TaskList: MarkdownConfig = {
|
|
|
|
defineNodes: [
|
|
|
|
{ name: "Task", block: true, style: t.list },
|
2023-10-03 20:16:33 +08:00
|
|
|
{ name: "TaskMark", style: t.atom },
|
2023-09-01 22:57:29 +08:00
|
|
|
{ name: "TaskState", style: TaskStateTag },
|
|
|
|
],
|
|
|
|
parseBlock: [{
|
|
|
|
name: "TaskList",
|
|
|
|
leaf(cx, leaf) {
|
|
|
|
const match = /^\[([^\]]+)\][ \t]/.exec(leaf.content);
|
|
|
|
return match &&
|
|
|
|
cx.parentType().name == "ListItem"
|
|
|
|
? new MultiStatusTaskParser(match[1])
|
|
|
|
: null;
|
|
|
|
},
|
|
|
|
after: "SetextHeading",
|
|
|
|
}],
|
|
|
|
};
|