2024-05-28 02:33:41 +08:00
|
|
|
import { isLocalPath, resolvePath } from "$sb/lib/resolve.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { Client } from "../client.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { syntaxTree } from "@codemirror/language";
|
|
|
|
import { Decoration } from "@codemirror/view";
|
2022-11-18 23:04:37 +08:00
|
|
|
import {
|
2022-12-09 23:09:53 +08:00
|
|
|
decoratorStateField,
|
2022-11-18 23:04:37 +08:00
|
|
|
invisibleDecoration,
|
2022-11-28 23:42:54 +08:00
|
|
|
isCursorInRange,
|
2022-11-18 23:04:37 +08:00
|
|
|
} from "./util.ts";
|
2022-11-28 23:42:54 +08:00
|
|
|
|
2023-12-20 00:55:11 +08:00
|
|
|
export function linkPlugin(client: Client) {
|
2022-12-09 23:09:53 +08:00
|
|
|
return decoratorStateField((state) => {
|
|
|
|
const widgets: any[] = [];
|
2022-11-28 23:42:54 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
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;
|
|
|
|
}
|
2022-11-29 16:17:40 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
const text = state.sliceDoc(from, to);
|
|
|
|
// Links are of the form [hell](https://example.com)
|
|
|
|
const [anchorPart, linkPart] = text.split("]("); // Not pretty
|
2022-12-29 19:53:42 +08:00
|
|
|
if (anchorPart.substring(1).trim() === "") {
|
|
|
|
// Empty link text, let's not do live preview (because it would make it disappear)
|
|
|
|
return;
|
|
|
|
}
|
2022-12-09 23:09:53 +08:00
|
|
|
if (!linkPart) {
|
|
|
|
// Invalid link
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const cleanAnchor = anchorPart.substring(1); // cut off the initial [
|
2023-12-20 00:55:11 +08:00
|
|
|
let cleanLink = linkPart.substring(0, linkPart.length - 1); // cut off the final )
|
|
|
|
|
2024-05-28 02:33:41 +08:00
|
|
|
if (isLocalPath(cleanLink)) {
|
|
|
|
cleanLink = resolvePath(
|
2024-01-24 21:44:39 +08:00
|
|
|
client.currentPage,
|
2023-12-20 00:55:11 +08:00
|
|
|
decodeURI(cleanLink),
|
|
|
|
);
|
|
|
|
}
|
2022-11-29 16:17:40 +08:00
|
|
|
|
2023-01-13 23:33:36 +08:00
|
|
|
// Hide the start [
|
2022-12-09 23:09:53 +08:00
|
|
|
widgets.push(
|
|
|
|
invisibleDecoration.range(
|
|
|
|
from,
|
2023-01-13 23:33:36 +08:00
|
|
|
from + 1,
|
2022-12-09 23:09:53 +08:00
|
|
|
),
|
|
|
|
);
|
2023-01-13 23:33:36 +08:00
|
|
|
// 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
|
2022-12-09 23:09:53 +08:00
|
|
|
widgets.push(
|
2023-01-13 23:33:36 +08:00
|
|
|
invisibleDecoration.range(
|
|
|
|
from + cleanAnchor.length + 1,
|
|
|
|
to,
|
|
|
|
),
|
2022-12-09 23:09:53 +08:00
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|
2022-11-18 23:04:37 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
return Decoration.set(widgets, true);
|
|
|
|
});
|
2022-11-18 23:04:37 +08:00
|
|
|
}
|