67 lines
1.9 KiB
TypeScript
67 lines
1.9 KiB
TypeScript
import { Decoration, syntaxTree } from "../deps.ts";
|
|
import {
|
|
decoratorStateField,
|
|
invisibleDecoration,
|
|
isCursorInRange,
|
|
} from "./util.ts";
|
|
|
|
export function linkPlugin() {
|
|
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 [
|
|
const cleanLink = linkPart.substring(0, linkPart.length - 1); // cut off the final )
|
|
|
|
// 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);
|
|
});
|
|
}
|