import { syntaxTree } from "@codemirror/language"; import { Decoration, WidgetType } from "@codemirror/view"; import { NodeType } from "@lezer/common"; import { decoratorStateField, isCursorInRange } from "./util.ts"; /** * Widget to render checkbox for a task list item. */ class CheckboxWidget extends WidgetType { constructor( public checked: boolean, readonly pos: number, readonly clickCallback: (pos: number) => void, ) { super(); } toDOM(): HTMLElement { const wrap = document.createElement("span"); wrap.classList.add("sb-checkbox"); const checkbox = document.createElement("input"); checkbox.type = "checkbox"; checkbox.checked = this.checked; checkbox.addEventListener("click", (e) => { e.stopPropagation(); e.preventDefault(); }); checkbox.addEventListener("mouseup", (e) => { e.stopPropagation(); this.clickCallback(this.pos); }); wrap.appendChild(checkbox); return wrap; } } export function taskListPlugin( { onCheckboxClick }: { onCheckboxClick: (pos: number) => void }, ) { return decoratorStateField((state) => { const widgets: any[] = []; syntaxTree(state).iterate({ enter({ type, from, to, node }) { if (type.name !== "Task") return; // true/false if this is a checkbox, undefined when it's a custom-status task let checkboxStatus: boolean | undefined; // Iterate inside the task node to find the checkbox node.toTree().iterate({ enter: (ref) => iterateInner(ref.type, ref.from, ref.to), }); if (checkboxStatus === true) { widgets.push( Decoration.mark({ tagName: "span", class: "cm-task-checked", }).range(from, to), ); } function iterateInner(type: NodeType, nfrom: number, nto: number) { if (type.name !== "TaskState") return; if (isCursorInRange(state, [from + nfrom, from + nto])) return; const checkbox = state.sliceDoc(from + nfrom, from + nto); // Checkbox is checked if it has a 'x' in between the [] if (checkbox === "[x]" || checkbox === "[X]") { checkboxStatus = true; } else if (checkbox === "[ ]") { checkboxStatus = false; } if (checkboxStatus === undefined) { // Not replacing it with a widget return; } const dec = Decoration.replace({ widget: new CheckboxWidget( checkboxStatus, from + nfrom + 1, onCheckboxClick, ), }); widgets.push(dec.range(from + nfrom, from + nto)); } }, }); return Decoration.set(widgets, true); }); }