silverbullet/web/cm_plugins/table.ts

132 lines
3.8 KiB
TypeScript
Raw Normal View History

2024-07-30 23:33:33 +08:00
import type { EditorState } from "@codemirror/state";
import { syntaxTree } from "@codemirror/language";
import { Decoration, WidgetType } from "@codemirror/view";
import {
decoratorStateField,
invisibleDecoration,
isCursorInRange,
} from "./util.ts";
import { renderMarkdownToHtml } from "../../plugs/markdown/markdown_render.ts";
2024-07-30 23:33:33 +08:00
import { type ParseTree, renderToText } from "../../plug-api/lib/tree.ts";
import { lezerToParseTree } from "$common/markdown_parser/parse_tree.ts";
2023-07-14 22:56:20 +08:00
import type { Client } from "../client.ts";
2024-05-28 02:33:41 +08:00
import { isLocalPath, resolvePath } from "$sb/lib/resolve.ts";
class TableViewWidget extends WidgetType {
2024-01-02 18:32:57 +08:00
tableBodyText: string;
constructor(
readonly pos: number,
2024-01-02 18:32:57 +08:00
readonly client: Client,
readonly t: ParseTree,
) {
super();
2024-01-02 18:32:57 +08:00
this.tableBodyText = renderToText(t);
}
toDOM(): HTMLElement {
const dom = document.createElement("span");
dom.classList.add("sb-table-widget");
dom.addEventListener("click", (e) => {
// Pulling data-pos to put the cursor in the right place, falling back
// to the start of the table.
const dataAttributes = (e.target as any).dataset;
2024-01-02 18:32:57 +08:00
this.client.editorView.dispatch({
selection: {
anchor: dataAttributes.pos ? +dataAttributes.pos : this.pos,
},
});
});
dom.innerHTML = renderMarkdownToHtml(this.t, {
// Annotate every element with its position so we can use it to put
// the cursor there when the user clicks on the table.
annotationPositions: true,
2023-12-20 00:55:11 +08:00
translateUrls: (url) => {
2024-05-28 02:33:41 +08:00
if (isLocalPath(url)) {
url = resolvePath(this.client.currentPage, decodeURI(url));
2023-02-23 22:33:51 +08:00
}
2023-12-20 00:20:47 +08:00
2023-02-23 22:33:51 +08:00
return url;
},
2023-07-25 01:54:31 +08:00
preserveAttributes: true,
});
2024-01-02 18:32:57 +08:00
setTimeout(() => {
this.client.setCachedWidgetHeight(
`table:${this.tableBodyText}`,
dom.clientHeight,
);
});
return dom;
}
2024-01-02 18:32:57 +08:00
get estimatedHeight(): number {
const height = this.client.getCachedWidgetHeight(
`table:${this.tableBodyText}`,
);
// console.log("Calling estimated height for table", height);
2024-01-02 18:32:57 +08:00
return height;
}
eq(other: WidgetType): boolean {
return (
other instanceof TableViewWidget &&
other.tableBodyText === this.tableBodyText
);
}
}
2023-07-14 22:56:20 +08:00
export function tablePlugin(editor: Client) {
return decoratorStateField((state: EditorState) => {
const widgets: any[] = [];
syntaxTree(state).iterate({
enter: (node) => {
const { from, to, name } = node;
if (name !== "Table") return;
if (isCursorInRange(state, [from, to])) return;
const tableText = state.sliceDoc(from, to);
const lineStrings = tableText.split("\n");
const lines: { from: number; to: number }[] = [];
let fromIt = from;
for (const line of lineStrings) {
lines.push({
from: fromIt,
to: fromIt + line.length,
});
fromIt += line.length + 1;
}
const firstLine = lines[0], lastLine = lines[lines.length - 1];
// In case of doubt, back out
if (!firstLine || !lastLine) return;
widgets.push(invisibleDecoration.range(firstLine.from, firstLine.to));
widgets.push(invisibleDecoration.range(lastLine.from, lastLine.to));
lines.slice(1, lines.length - 1).forEach((line) => {
widgets.push(
Decoration.line({ class: "sb-line-table-outside" }).range(
line.from,
),
);
});
const text = state.sliceDoc(0, to);
widgets.push(
Decoration.widget({
widget: new TableViewWidget(
from,
2023-02-23 22:33:51 +08:00
editor,
lezerToParseTree(text, node.node),
),
}).range(from),
);
},
});
return Decoration.set(widgets, true);
});
}