silverbullet/web/cm_plugins/link.ts

77 lines
2.2 KiB
TypeScript

import { isLocalPath, resolvePath } from "$sb/lib/resolve.ts";
import type { Client } from "../client.ts";
import { syntaxTree } from "@codemirror/language";
import { Decoration } from "@codemirror/view";
import {
decoratorStateField,
invisibleDecoration,
isCursorInRange,
} from "./util.ts";
export function linkPlugin(client: Client) {
return decoratorStateField((state) => {
const widgets: any[] = [];
syntaxTree(state).iterate({
enter: ({ type, from, to }) => {
if (type.name !== "Link") {
return;
}
// Adding 2 on each side due to [[ and ]] that are outside the WikiLinkPage node
if (isCursorInRange(state, [from, to])) {
return;
}
const text = state.sliceDoc(from, to);
// Links are of the form [hell](https://example.com)
const [anchorPart, linkPart] = text.split("]("); // Not pretty
if (anchorPart.substring(1).trim() === "") {
// Empty link text, let's not do live preview (because it would make it disappear)
return;
}
if (!linkPart) {
// Invalid link
return;
}
const cleanAnchor = anchorPart.substring(1); // cut off the initial [
let cleanLink = linkPart.substring(0, linkPart.length - 1); // cut off the final )
if (isLocalPath(cleanLink)) {
cleanLink = resolvePath(
client.currentPage,
decodeURI(cleanLink),
);
}
// Hide the start [
widgets.push(
invisibleDecoration.range(
from,
from + 1,
),
);
// Wrap the link in a href
widgets.push(
Decoration.mark({
tagName: "a",
class: "sb-link",
attributes: {
href: cleanLink,
title: `Click to visit ${cleanLink}`,
},
}).range(from + 1, from + cleanAnchor.length + 1),
);
// Hide the tail end of the link
widgets.push(
invisibleDecoration.range(
from + cleanAnchor.length + 1,
to,
),
);
},
});
return Decoration.set(widgets, true);
});
}