1
0
silverbullet/plugs/tasks/task.ts

127 lines
3.3 KiB
TypeScript
Raw Normal View History

2022-03-28 13:25:05 +00:00
import type { ClickEvent } from "../../webapp/app_event";
import { IndexEvent } from "../../webapp/app_event";
2022-03-29 15:02:28 +00:00
import { whiteOutQueries } from "../core/materialized_queries";
2022-04-01 15:07:08 +00:00
import { batchSet, scanPrefixGlobal } from "plugos-silverbullet-syscall/index";
import { readPage, writePage } from "plugos-silverbullet-syscall/space";
import {
dispatch,
getLineUnderCursor,
getSyntaxNodeAtPos,
} from "plugos-silverbullet-syscall/editor";
2022-03-29 15:02:28 +00:00
2022-03-28 13:25:05 +00:00
const allTasksPageName = "ALL TASKS";
const taskRe = /[\-\*]\s*\[([ Xx])\]\s*(.*)/g;
2022-03-29 15:02:28 +00:00
const taskFullRe =
/(?<prefix>[\t ]*)[\-\*]\s*\[([ Xx])\]\s*([^\n]+)(\n\k<prefix>\s+[\-\*][^\n]+)*/g;
2022-03-28 13:25:05 +00:00
const extractPageLink = /[\-\*]\s*\[[ Xx]\]\s\[\[([^\]]+)@(\d+)\]\]\s*(.*)/;
2022-03-29 15:02:28 +00:00
type Task = {
task: string;
complete: boolean;
pos?: number;
children?: string[];
};
2022-03-28 13:25:05 +00:00
export async function indexTasks({ name, text }: IndexEvent) {
if (name === allTasksPageName) {
return;
}
console.log("Indexing tasks");
let tasks: { key: string; value: Task }[] = [];
2022-03-29 15:02:28 +00:00
text = whiteOutQueries(text);
for (let match of text.matchAll(taskFullRe)) {
let entire = match[0];
let complete = match[2] !== " ";
let task = match[3];
2022-03-28 13:25:05 +00:00
let pos = match.index!;
2022-03-29 15:02:28 +00:00
let lines = entire.split("\n");
let value: Task = {
task,
complete,
};
if (lines.length > 1) {
value.children = lines.slice(1);
}
2022-03-28 13:25:05 +00:00
tasks.push({
key: `task:${pos}`,
2022-03-29 15:02:28 +00:00
value,
2022-03-28 13:25:05 +00:00
});
}
console.log("Found", tasks.length, "task(s)");
2022-04-01 15:07:08 +00:00
await batchSet(name, tasks);
2022-03-28 13:25:05 +00:00
}
export async function updateTaskPage() {
2022-04-01 15:07:08 +00:00
let allTasks = await scanPrefixGlobal("task:");
2022-03-28 13:25:05 +00:00
let pageTasks = new Map<string, Task[]>();
for (let {
key,
page,
2022-03-29 15:02:28 +00:00
value: { task, complete },
2022-03-28 13:25:05 +00:00
} of allTasks) {
if (complete) {
continue;
}
let [, pos] = key.split(":");
let tasks = pageTasks.get(page) || [];
2022-04-01 15:07:08 +00:00
tasks.push({ task, complete, pos: +pos });
2022-03-28 13:25:05 +00:00
pageTasks.set(page, tasks);
}
let mdPieces = [];
for (let pageName of [...pageTasks.keys()].sort()) {
mdPieces.push(`\n## ${pageName}\n`);
for (let task of pageTasks.get(pageName)!) {
mdPieces.push(
`* [${task.complete ? "x" : " "}] [[${pageName}@${task.pos}]] ${
task.task
}`
);
}
}
let taskMd = mdPieces.join("\n");
2022-04-01 15:07:08 +00:00
await writePage(allTasksPageName, taskMd);
2022-03-28 13:25:05 +00:00
}
export async function taskToggle(event: ClickEvent) {
2022-04-01 15:07:08 +00:00
let syntaxNode = await getSyntaxNodeAtPos(event.pos);
2022-03-28 13:25:05 +00:00
if (syntaxNode && syntaxNode.name === "TaskMarker") {
let changeTo = "[x]";
if (syntaxNode.text === "[x]" || syntaxNode.text === "[X]") {
changeTo = "[ ]";
}
2022-04-01 15:07:08 +00:00
await dispatch({
2022-03-28 13:25:05 +00:00
changes: {
from: syntaxNode.from,
to: syntaxNode.to,
insert: changeTo,
},
selection: {
anchor: event.pos,
},
});
if (event.page === allTasksPageName) {
// Propagate back to the page in question
2022-04-01 15:07:08 +00:00
let line = await getLineUnderCursor();
2022-03-28 13:25:05 +00:00
let match = line.match(extractPageLink);
if (match) {
let [, page, posS] = match;
let pos = +posS;
2022-04-01 15:07:08 +00:00
let pageData = await readPage(page);
2022-03-28 13:25:05 +00:00
let text = pageData.text;
// Apply the toggle
text =
text.substring(0, pos) +
text.substring(pos).replace(/^([\-\*]\s*)\[[ xX]\]/, "$1" + changeTo);
2022-04-01 15:07:08 +00:00
await writePage(page, text);
2022-03-28 13:25:05 +00:00
}
}
}
}