2024-02-29 22:23:05 +08:00
|
|
|
import type { ClickEvent, IndexTreeEvent } from "../../plug-api/types.ts";
|
2022-03-28 21:25:05 +08:00
|
|
|
|
2024-08-07 02:11:38 +08:00
|
|
|
import {
|
|
|
|
editor,
|
|
|
|
events,
|
|
|
|
markdown,
|
|
|
|
space,
|
|
|
|
sync,
|
|
|
|
} from "@silverbulletmd/silverbullet/syscalls";
|
2022-10-14 21:11:33 +08:00
|
|
|
|
2022-04-12 02:34:09 +08:00
|
|
|
import {
|
|
|
|
addParentPointers,
|
|
|
|
collectNodesMatching,
|
2024-01-26 18:10:35 +08:00
|
|
|
findNodeMatching,
|
2022-04-21 17:46:33 +08:00
|
|
|
findNodeOfType,
|
2023-09-01 22:57:29 +08:00
|
|
|
findParentMatching,
|
2022-04-12 02:34:09 +08:00
|
|
|
nodeAtPos,
|
2024-07-30 23:33:33 +08:00
|
|
|
type ParseTree,
|
2022-04-25 16:33:38 +08:00
|
|
|
renderToText,
|
2022-07-04 17:30:30 +08:00
|
|
|
replaceNodesMatching,
|
2023-07-26 23:12:56 +08:00
|
|
|
traverseTreeAsync,
|
2024-02-29 22:23:05 +08:00
|
|
|
} from "../../plug-api/lib/tree.ts";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { niceDate } from "$lib/dates.ts";
|
2024-08-07 02:11:38 +08:00
|
|
|
import { extractAttributes } from "@silverbulletmd/silverbullet/lib/attribute";
|
|
|
|
import { rewritePageRefs } from "@silverbulletmd/silverbullet/lib/resolve";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { ObjectValue } from "../../plug-api/types.ts";
|
2023-10-03 20:16:33 +08:00
|
|
|
import { indexObjects, queryObjects } from "../index/plug_api.ts";
|
2024-08-07 02:11:38 +08:00
|
|
|
import { updateITags } from "@silverbulletmd/silverbullet/lib/tags";
|
|
|
|
import { extractFrontmatter } from "@silverbulletmd/silverbullet/lib/frontmatter";
|
|
|
|
import {
|
|
|
|
parsePageRef,
|
|
|
|
positionOfLine,
|
|
|
|
} from "@silverbulletmd/silverbullet/lib/page_ref";
|
2022-03-28 21:25:05 +08:00
|
|
|
|
2023-10-13 23:10:57 +08:00
|
|
|
export type TaskObject = ObjectValue<
|
|
|
|
{
|
|
|
|
page: string;
|
|
|
|
pos: number;
|
|
|
|
name: string;
|
2024-07-07 02:52:27 +08:00
|
|
|
text: string;
|
2023-10-13 23:10:57 +08:00
|
|
|
done: boolean;
|
|
|
|
state: string;
|
|
|
|
deadline?: string;
|
|
|
|
} & Record<string, any>
|
|
|
|
>;
|
2022-03-28 21:25:05 +08:00
|
|
|
|
2023-10-13 23:10:57 +08:00
|
|
|
export type TaskStateObject = ObjectValue<{
|
2023-10-03 20:16:33 +08:00
|
|
|
state: string;
|
|
|
|
count: number;
|
|
|
|
page: string;
|
2023-10-13 23:10:57 +08:00
|
|
|
}>;
|
2023-10-03 20:16:33 +08:00
|
|
|
|
2022-04-21 17:46:33 +08:00
|
|
|
function getDeadline(deadlineNode: ParseTree): string {
|
|
|
|
return deadlineNode.children![0].text!.replace(/📅\s*/, "");
|
|
|
|
}
|
|
|
|
|
2023-09-01 22:57:29 +08:00
|
|
|
const completeStates = ["x", "X"];
|
|
|
|
const incompleteStates = [" "];
|
|
|
|
|
2022-04-20 16:56:43 +08:00
|
|
|
export async function indexTasks({ name, tree }: IndexTreeEvent) {
|
2023-10-03 20:16:33 +08:00
|
|
|
const tasks: ObjectValue<TaskObject>[] = [];
|
|
|
|
const taskStates = new Map<string, { count: number; firstPos: number }>();
|
2024-01-11 20:20:50 +08:00
|
|
|
const frontmatter = await extractFrontmatter(tree);
|
|
|
|
|
2023-07-26 23:12:56 +08:00
|
|
|
await traverseTreeAsync(tree, async (n) => {
|
|
|
|
if (n.type !== "Task") {
|
|
|
|
return false;
|
|
|
|
}
|
2023-09-01 22:57:29 +08:00
|
|
|
const state = n.children![0].children![1].text!;
|
|
|
|
if (!incompleteStates.includes(state) && !completeStates.includes(state)) {
|
2023-10-03 20:16:33 +08:00
|
|
|
let currentState = taskStates.get(state);
|
|
|
|
if (!currentState) {
|
|
|
|
currentState = { count: 0, firstPos: n.from! };
|
2023-10-13 23:03:20 +08:00
|
|
|
taskStates.set(state, currentState);
|
2023-09-01 22:57:29 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
currentState.count++;
|
2023-09-01 22:57:29 +08:00
|
|
|
}
|
|
|
|
const complete = completeStates.includes(state);
|
2023-10-03 20:16:33 +08:00
|
|
|
const task: TaskObject = {
|
|
|
|
ref: `${name}@${n.from}`,
|
2024-01-11 20:20:50 +08:00
|
|
|
tag: "task",
|
2022-07-04 17:30:30 +08:00
|
|
|
name: "",
|
2024-07-07 02:52:27 +08:00
|
|
|
text: "",
|
2022-04-12 19:33:07 +08:00
|
|
|
done: complete,
|
2023-10-03 20:16:33 +08:00
|
|
|
page: name,
|
|
|
|
pos: n.from!,
|
2023-09-01 22:57:29 +08:00
|
|
|
state,
|
2022-03-29 23:02:28 +08:00
|
|
|
};
|
2022-04-12 02:34:09 +08:00
|
|
|
|
2023-07-31 01:31:04 +08:00
|
|
|
rewritePageRefs(n, name);
|
|
|
|
|
2024-07-07 02:52:27 +08:00
|
|
|
// The task text is everything after the task marker
|
|
|
|
task.text = n.children!.slice(1).map(renderToText).join("").trim();
|
|
|
|
|
|
|
|
// This finds the deadline and tags, and removes them from the tree
|
2022-07-04 17:30:30 +08:00
|
|
|
replaceNodesMatching(n, (tree) => {
|
|
|
|
if (tree.type === "DeadlineDate") {
|
|
|
|
task.deadline = getDeadline(tree);
|
|
|
|
// Remove this node from the tree
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (tree.type === "Hashtag") {
|
2022-11-20 17:24:24 +08:00
|
|
|
// Push the tag to the list, removing the initial #
|
2023-10-03 20:16:33 +08:00
|
|
|
const tagName = tree.children![0].text!.substring(1);
|
2024-01-11 20:20:50 +08:00
|
|
|
if (!task.tags) {
|
|
|
|
task.tags = [];
|
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
task.tags.push(tagName);
|
2024-02-28 03:05:12 +08:00
|
|
|
tree.children = [];
|
2022-07-04 17:30:30 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-07-25 01:54:31 +08:00
|
|
|
// Extract attributes and remove from tree
|
2024-02-28 03:05:12 +08:00
|
|
|
const extractedAttributes = await extractAttributes(
|
|
|
|
["task", ...task.tags || []],
|
|
|
|
n,
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
task.name = n.children!.slice(1).map(renderToText).join("").trim();
|
|
|
|
|
2023-07-25 01:54:31 +08:00
|
|
|
for (const [key, value] of Object.entries(extractedAttributes)) {
|
|
|
|
task[key] = value;
|
|
|
|
}
|
|
|
|
|
2024-01-11 20:20:50 +08:00
|
|
|
updateITags(task, frontmatter);
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
tasks.push(task);
|
2023-07-26 23:12:56 +08:00
|
|
|
return true;
|
2022-04-04 17:51:41 +08:00
|
|
|
});
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
// Index task states
|
|
|
|
if (taskStates.size > 0) {
|
|
|
|
await indexObjects<TaskStateObject>(
|
|
|
|
name,
|
|
|
|
Array.from(taskStates.entries()).map(([state, { firstPos, count }]) => ({
|
|
|
|
ref: `${name}@${firstPos}`,
|
2024-01-11 20:20:50 +08:00
|
|
|
tag: "taskstate",
|
2023-10-03 20:16:33 +08:00
|
|
|
state,
|
|
|
|
count,
|
|
|
|
page: name,
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Index tasks themselves
|
|
|
|
if (tasks.length > 0) {
|
|
|
|
await indexObjects(name, tasks);
|
|
|
|
}
|
2022-03-28 21:25:05 +08:00
|
|
|
}
|
|
|
|
|
2022-10-14 21:11:33 +08:00
|
|
|
export function taskToggle(event: ClickEvent) {
|
2023-09-01 22:57:29 +08:00
|
|
|
if (event.altKey) {
|
|
|
|
return;
|
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
return taskCycleAtPos(event.pos);
|
2022-03-28 21:25:05 +08:00
|
|
|
}
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
export function previewTaskToggle(eventString: string) {
|
2022-11-01 22:01:28 +08:00
|
|
|
const [eventName, pos] = JSON.parse(eventString);
|
|
|
|
if (eventName === "task") {
|
2023-10-03 20:16:33 +08:00
|
|
|
return taskCycleAtPos(+pos);
|
2022-11-01 22:01:28 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-01 22:57:29 +08:00
|
|
|
async function cycleTaskState(
|
2023-07-02 17:25:32 +08:00
|
|
|
node: ParseTree,
|
|
|
|
) {
|
2023-09-01 22:57:29 +08:00
|
|
|
const stateText = node.children![1].text!;
|
|
|
|
let changeTo: string | undefined;
|
|
|
|
if (completeStates.includes(stateText)) {
|
|
|
|
changeTo = " ";
|
|
|
|
} else if (incompleteStates.includes(stateText)) {
|
|
|
|
changeTo = "x";
|
|
|
|
} else {
|
|
|
|
// Not a checkbox, but a custom state
|
2023-10-03 20:16:33 +08:00
|
|
|
const allStates = await queryObjects<TaskStateObject>("taskstate", {});
|
|
|
|
const states = [...new Set(allStates.map((s) => s.state))];
|
2023-09-01 22:57:29 +08:00
|
|
|
states.sort();
|
|
|
|
// Select a next state
|
|
|
|
const currentStateIndex = states.indexOf(stateText);
|
|
|
|
if (currentStateIndex === -1) {
|
|
|
|
console.error("Unknown state", stateText);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const nextStateIndex = (currentStateIndex + 1) % states.length;
|
|
|
|
changeTo = states[nextStateIndex];
|
|
|
|
// console.log("All possible states", states);
|
|
|
|
// return;
|
2022-04-21 17:46:33 +08:00
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
await editor.dispatch({
|
2022-04-21 17:46:33 +08:00
|
|
|
changes: {
|
2023-09-01 22:57:29 +08:00
|
|
|
from: node.children![1].from,
|
|
|
|
to: node.children![1].to,
|
2022-04-21 17:46:33 +08:00
|
|
|
insert: changeTo,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-10-14 21:11:33 +08:00
|
|
|
const parentWikiLinks = collectNodesMatching(
|
2022-04-21 17:46:33 +08:00
|
|
|
node.parent!,
|
2022-10-10 20:50:21 +08:00
|
|
|
(n) => n.type === "WikiLinkPage",
|
2022-04-21 17:46:33 +08:00
|
|
|
);
|
2022-10-14 21:11:33 +08:00
|
|
|
for (const wikiLink of parentWikiLinks) {
|
|
|
|
const ref = wikiLink.children![0].text!;
|
2022-04-21 17:46:33 +08:00
|
|
|
if (ref.includes("@")) {
|
2023-10-03 20:16:33 +08:00
|
|
|
await updateTaskState(ref, stateText, changeTo);
|
|
|
|
}
|
|
|
|
}
|
2024-02-28 17:55:08 +08:00
|
|
|
|
|
|
|
await events.dispatchEvent("task:stateChange", {
|
|
|
|
from: node.parent!.from,
|
|
|
|
to: node.parent!.to,
|
|
|
|
newState: changeTo,
|
|
|
|
text: renderToText(node.parent),
|
|
|
|
});
|
2023-10-03 20:16:33 +08:00
|
|
|
}
|
2023-07-28 19:54:31 +08:00
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
export async function updateTaskState(
|
|
|
|
ref: string,
|
|
|
|
oldState: string,
|
|
|
|
newState: string,
|
|
|
|
) {
|
|
|
|
const currentPage = await editor.getCurrentPage();
|
2024-01-24 18:58:33 +08:00
|
|
|
const { page, pos } = parsePageRef(ref);
|
|
|
|
if (pos === undefined) {
|
|
|
|
console.error("No position found in page ref, skipping", ref);
|
|
|
|
return;
|
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
if (page === currentPage) {
|
|
|
|
// In current page, just update the task marker with dispatch
|
|
|
|
const editorText = await editor.getText();
|
2024-07-29 02:31:37 +08:00
|
|
|
const targetPos = pos instanceof Object
|
|
|
|
? positionOfLine(editorText, pos.line, pos.column)
|
|
|
|
: pos;
|
2023-10-03 20:16:33 +08:00
|
|
|
// Check if the task state marker is still there
|
|
|
|
const targetText = editorText.substring(
|
2024-07-29 02:31:37 +08:00
|
|
|
targetPos + 1,
|
|
|
|
targetPos + 1 + oldState.length,
|
2023-10-03 20:16:33 +08:00
|
|
|
);
|
|
|
|
if (targetText !== oldState) {
|
|
|
|
console.error(
|
|
|
|
"Reference not a task marker, out of date?",
|
|
|
|
targetText,
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
await editor.dispatch({
|
|
|
|
changes: {
|
2024-07-29 02:31:37 +08:00
|
|
|
from: targetPos + 1,
|
|
|
|
to: targetPos + 1 + oldState.length,
|
2023-10-03 20:16:33 +08:00
|
|
|
insert: newState,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
let text = await space.readPage(page);
|
|
|
|
|
|
|
|
const referenceMdTree = await markdown.parseMarkdown(text);
|
2024-07-29 02:31:37 +08:00
|
|
|
const targetPos = pos instanceof Object
|
|
|
|
? positionOfLine(text, pos.line, pos.column)
|
|
|
|
: pos;
|
2023-10-03 20:16:33 +08:00
|
|
|
// Adding +1 to immediately hit the task state node
|
2024-07-29 02:31:37 +08:00
|
|
|
const taskStateNode = nodeAtPos(referenceMdTree, targetPos + 1);
|
2023-10-03 20:16:33 +08:00
|
|
|
if (!taskStateNode || taskStateNode.type !== "TaskState") {
|
|
|
|
console.error(
|
|
|
|
"Reference not a task marker, out of date?",
|
|
|
|
taskStateNode,
|
|
|
|
);
|
|
|
|
return;
|
2022-04-21 17:46:33 +08:00
|
|
|
}
|
2023-10-03 20:16:33 +08:00
|
|
|
taskStateNode.children![1].text = newState;
|
|
|
|
text = renderToText(referenceMdTree);
|
|
|
|
await space.writePage(page, text);
|
|
|
|
sync.scheduleFileSync(`${page}.md`);
|
2022-04-21 17:46:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
export async function taskCycleAtPos(pos: number) {
|
2022-10-14 21:11:33 +08:00
|
|
|
const text = await editor.getText();
|
|
|
|
const mdTree = await markdown.parseMarkdown(text);
|
2022-04-04 17:51:41 +08:00
|
|
|
addParentPointers(mdTree);
|
|
|
|
|
2023-09-01 22:57:29 +08:00
|
|
|
let node = nodeAtPos(mdTree, pos);
|
|
|
|
if (node) {
|
2023-10-03 20:16:33 +08:00
|
|
|
if (node.type === "TaskMark") {
|
2023-09-01 22:57:29 +08:00
|
|
|
node = node.parent!;
|
|
|
|
}
|
|
|
|
if (node.type === "TaskState") {
|
2023-10-03 20:16:33 +08:00
|
|
|
await cycleTaskState(node);
|
2023-09-01 22:57:29 +08:00
|
|
|
}
|
2022-04-21 17:46:33 +08:00
|
|
|
}
|
|
|
|
}
|
2022-03-28 21:25:05 +08:00
|
|
|
|
2023-09-01 22:57:29 +08:00
|
|
|
export async function taskCycleCommand() {
|
2022-10-14 21:11:33 +08:00
|
|
|
const text = await editor.getText();
|
|
|
|
const pos = await editor.getCursor();
|
|
|
|
const tree = await markdown.parseMarkdown(text);
|
2022-04-21 17:46:33 +08:00
|
|
|
addParentPointers(tree);
|
|
|
|
|
2023-09-01 22:57:29 +08:00
|
|
|
let node = nodeAtPos(tree, pos);
|
|
|
|
if (!node) {
|
|
|
|
await editor.flashNotification("No task at cursor");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (["BulletList", "Document"].includes(node.type!)) {
|
|
|
|
// Likely at the end of the line, let's back up a position
|
|
|
|
node = nodeAtPos(tree, pos - 1);
|
|
|
|
}
|
|
|
|
if (!node) {
|
|
|
|
await editor.flashNotification("No task at cursor");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
console.log("Node", node);
|
|
|
|
const taskNode = node.type === "Task"
|
|
|
|
? node
|
|
|
|
: findParentMatching(node!, (n) => n.type === "Task");
|
|
|
|
if (!taskNode) {
|
|
|
|
await editor.flashNotification("No task at cursor");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const taskState = findNodeOfType(taskNode!, "TaskState");
|
|
|
|
if (taskState) {
|
2023-10-03 20:16:33 +08:00
|
|
|
await cycleTaskState(taskState);
|
2023-09-01 22:57:29 +08:00
|
|
|
}
|
2022-04-21 17:46:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function postponeCommand() {
|
2022-10-14 21:11:33 +08:00
|
|
|
const text = await editor.getText();
|
|
|
|
const pos = await editor.getCursor();
|
|
|
|
const tree = await markdown.parseMarkdown(text);
|
2022-04-21 17:46:33 +08:00
|
|
|
addParentPointers(tree);
|
|
|
|
|
2022-10-14 21:11:33 +08:00
|
|
|
const node = nodeAtPos(tree, pos)!;
|
2022-04-21 17:46:33 +08:00
|
|
|
// We kwow node.type === DeadlineDate (due to the task context)
|
2022-10-14 21:11:33 +08:00
|
|
|
const date = getDeadline(node);
|
|
|
|
const option = await editor.filterBox(
|
2022-04-21 17:46:33 +08:00
|
|
|
"Postpone for...",
|
|
|
|
[
|
|
|
|
{ name: "a day", orderId: 1 },
|
|
|
|
{ name: "a week", orderId: 2 },
|
|
|
|
{ name: "following Monday", orderId: 3 },
|
|
|
|
],
|
2022-10-10 20:50:21 +08:00
|
|
|
"Select the desired time span to delay this task",
|
2022-04-21 17:46:33 +08:00
|
|
|
);
|
|
|
|
if (!option) {
|
|
|
|
return;
|
|
|
|
}
|
2023-05-22 17:54:25 +08:00
|
|
|
// Parse "naive" due date
|
2023-09-01 22:57:29 +08:00
|
|
|
const [yyyy, mm, dd] = date.split("-").map(Number);
|
2023-05-22 17:54:25 +08:00
|
|
|
// Create new naive Date object.
|
|
|
|
// `monthIndex` parameter is zero-based, so subtract 1 from parsed month.
|
|
|
|
const d = new Date(yyyy, mm - 1, dd);
|
2022-04-21 17:46:33 +08:00
|
|
|
switch (option.name) {
|
|
|
|
case "a day":
|
|
|
|
d.setDate(d.getDate() + 1);
|
|
|
|
break;
|
|
|
|
case "a week":
|
|
|
|
d.setDate(d.getDate() + 7);
|
|
|
|
break;
|
|
|
|
case "following Monday":
|
|
|
|
d.setDate(d.getDate() + ((7 - d.getDay() + 1) % 7 || 7));
|
|
|
|
break;
|
2022-03-28 21:25:05 +08:00
|
|
|
}
|
2023-05-22 17:54:25 +08:00
|
|
|
// console.log("New date", niceDate(d));
|
2022-10-14 21:11:33 +08:00
|
|
|
await editor.dispatch({
|
2022-04-21 17:46:33 +08:00
|
|
|
changes: {
|
|
|
|
from: node.from,
|
|
|
|
to: node.to,
|
|
|
|
insert: `📅 ${niceDate(d)}`,
|
|
|
|
},
|
|
|
|
selection: {
|
|
|
|
anchor: pos,
|
|
|
|
},
|
|
|
|
});
|
2022-03-28 21:25:05 +08:00
|
|
|
}
|
2024-01-26 18:10:35 +08:00
|
|
|
|
|
|
|
export async function removeCompletedTasksCommand() {
|
|
|
|
const tree = await markdown.parseMarkdown(await editor.getText());
|
|
|
|
addParentPointers(tree);
|
|
|
|
|
|
|
|
// Taking this ugly approach because the tree is modified in place
|
|
|
|
// Just finding and removing one task at a time and then repeating until nothing changes
|
|
|
|
while (true) {
|
|
|
|
const completedTaskNode = findNodeMatching(tree, (node) => {
|
|
|
|
return node.type === "Task" &&
|
|
|
|
["x", "X"].includes(node.children![0].children![1].text!);
|
|
|
|
});
|
|
|
|
if (completedTaskNode) {
|
|
|
|
// Ok got one, let's remove it
|
|
|
|
const listItemNode = completedTaskNode.parent!;
|
|
|
|
const bulletListNode = listItemNode.parent!;
|
|
|
|
// Remove the list item
|
|
|
|
const listItemIdx = bulletListNode.children!.indexOf(listItemNode);
|
|
|
|
let removeItems = 1;
|
|
|
|
if (bulletListNode.children![listItemIdx + 1]?.text === "\n") {
|
|
|
|
removeItems++;
|
|
|
|
}
|
|
|
|
bulletListNode.children!.splice(listItemIdx, removeItems);
|
|
|
|
} else {
|
|
|
|
// No completed tasks left, we're done
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
await editor.setText(renderToText(tree));
|
|
|
|
}
|