2024-03-16 22:29:24 +08:00
|
|
|
import { EditorState, Range } from "@codemirror/state";
|
|
|
|
import { syntaxTree } from "@codemirror/language";
|
|
|
|
import { Decoration, WidgetType } from "@codemirror/view";
|
2022-12-09 23:09:53 +08:00
|
|
|
import { decoratorStateField } from "./util.ts";
|
2023-07-14 22:56:20 +08:00
|
|
|
import type { Client } from "../client.ts";
|
2023-12-20 00:55:11 +08:00
|
|
|
import { resolveAttachmentPath, resolvePath } from "$sb/lib/resolve.ts";
|
2023-01-08 19:24:12 +08:00
|
|
|
|
2022-08-23 14:12:24 +08:00
|
|
|
class InlineImageWidget extends WidgetType {
|
2023-01-08 19:24:12 +08:00
|
|
|
constructor(
|
|
|
|
readonly url: string,
|
|
|
|
readonly title: string,
|
2023-07-30 05:41:37 +08:00
|
|
|
readonly client: Client,
|
2023-01-08 19:24:12 +08:00
|
|
|
) {
|
2022-08-23 14:12:24 +08:00
|
|
|
super();
|
2023-05-29 16:26:56 +08:00
|
|
|
// console.log("Creating widget", url);
|
2022-08-23 14:12:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
eq(other: InlineImageWidget) {
|
2022-08-24 00:53:05 +08:00
|
|
|
return other.url === this.url && other.title === this.title;
|
2022-08-23 14:12:24 +08:00
|
|
|
}
|
|
|
|
|
2023-05-29 16:26:56 +08:00
|
|
|
get estimatedHeight(): number {
|
2024-01-02 18:32:57 +08:00
|
|
|
const cachedHeight = this.client.getCachedWidgetHeight(`image:${this.url}`);
|
2023-05-29 16:26:56 +08:00
|
|
|
// console.log("Estimated height requested", this.url, cachedHeight);
|
|
|
|
return cachedHeight;
|
|
|
|
}
|
|
|
|
|
2022-08-23 14:12:24 +08:00
|
|
|
toDOM() {
|
2022-08-29 21:47:16 +08:00
|
|
|
const img = document.createElement("img");
|
2023-07-30 05:41:37 +08:00
|
|
|
let url = this.url;
|
2024-01-24 21:44:39 +08:00
|
|
|
url = resolvePath(this.client.currentPage, url, true);
|
2023-05-29 16:26:56 +08:00
|
|
|
// console.log("Creating DOM", this.url);
|
2024-01-02 18:32:57 +08:00
|
|
|
const cachedImageHeight = this.client.getCachedWidgetHeight(
|
|
|
|
`image:${this.url}`,
|
|
|
|
);
|
2023-05-29 16:26:56 +08:00
|
|
|
img.onload = () => {
|
|
|
|
// console.log("Loaded", this.url, "with height", img.height);
|
|
|
|
if (img.height !== cachedImageHeight) {
|
2024-01-02 18:32:57 +08:00
|
|
|
this.client.setCachedWidgetHeight(`image:${this.url}`, img.height);
|
2023-05-29 16:26:56 +08:00
|
|
|
}
|
|
|
|
};
|
2023-07-30 05:41:37 +08:00
|
|
|
img.src = url;
|
2022-08-24 00:53:05 +08:00
|
|
|
img.alt = this.title;
|
|
|
|
img.title = this.title;
|
2022-08-29 21:47:16 +08:00
|
|
|
img.style.display = "block";
|
|
|
|
img.className = "sb-inline-img";
|
2023-05-29 16:26:56 +08:00
|
|
|
if (cachedImageHeight > 0) {
|
|
|
|
img.height = cachedImageHeight;
|
|
|
|
}
|
2022-08-23 14:12:24 +08:00
|
|
|
|
|
|
|
return img;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-30 05:41:37 +08:00
|
|
|
export function inlineImagesPlugin(client: Client) {
|
2022-12-09 23:09:53 +08:00
|
|
|
return decoratorStateField((state: EditorState) => {
|
|
|
|
const widgets: Range<Decoration>[] = [];
|
|
|
|
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
|
2022-08-23 14:12:24 +08:00
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
syntaxTree(state).iterate({
|
2022-08-23 14:12:24 +08:00
|
|
|
enter: (node) => {
|
2022-08-29 21:47:16 +08:00
|
|
|
if (node.name !== "Image") {
|
|
|
|
return;
|
2022-08-23 14:12:24 +08:00
|
|
|
}
|
2022-08-29 21:47:16 +08:00
|
|
|
|
|
|
|
const imageRexexResult = imageRegex.exec(
|
2022-12-09 23:09:53 +08:00
|
|
|
state.sliceDoc(node.from, node.to),
|
2022-08-29 21:47:16 +08:00
|
|
|
);
|
2022-08-23 14:12:24 +08:00
|
|
|
if (imageRexexResult === null || !imageRexexResult.groups) {
|
|
|
|
return;
|
|
|
|
}
|
2022-08-29 21:47:16 +08:00
|
|
|
|
2023-07-02 17:25:32 +08:00
|
|
|
let url = imageRexexResult.groups.url;
|
2022-08-24 00:53:05 +08:00
|
|
|
const title = imageRexexResult.groups.title;
|
2023-12-20 00:20:47 +08:00
|
|
|
|
|
|
|
if (url.indexOf("://") === -1 && !url.startsWith("/")) {
|
2024-01-24 21:44:39 +08:00
|
|
|
url = resolveAttachmentPath(client.currentPage, decodeURI(url));
|
2023-07-02 17:25:32 +08:00
|
|
|
}
|
2022-11-18 23:04:37 +08:00
|
|
|
widgets.push(
|
|
|
|
Decoration.widget({
|
2023-07-30 05:41:37 +08:00
|
|
|
widget: new InlineImageWidget(url, title, client),
|
2023-02-28 17:07:20 +08:00
|
|
|
block: true,
|
2022-11-18 23:04:37 +08:00
|
|
|
}).range(node.to),
|
|
|
|
);
|
2022-08-29 21:47:16 +08:00
|
|
|
},
|
2022-08-23 14:12:24 +08:00
|
|
|
});
|
|
|
|
|
2022-12-09 23:09:53 +08:00
|
|
|
return Decoration.set(widgets, true);
|
|
|
|
});
|
|
|
|
}
|