silverbullet/web/cm_plugins/inline_image.ts

66 lines
1.5 KiB
TypeScript
Raw Normal View History

2022-08-29 21:47:16 +08:00
import {
Decoration,
EditorState,
Range,
syntaxTree,
2022-08-29 21:47:16 +08:00
WidgetType,
} from "../deps.ts";
import { decoratorStateField } from "./util.ts";
2022-08-23 14:12:24 +08:00
class InlineImageWidget extends WidgetType {
constructor(readonly url: string, readonly title: string) {
2022-08-23 14:12:24 +08:00
super();
}
eq(other: InlineImageWidget) {
return other.url === this.url && other.title === this.title;
2022-08-23 14:12:24 +08:00
}
toDOM() {
2022-08-29 21:47:16 +08:00
const img = document.createElement("img");
2022-09-06 20:36:06 +08:00
if (this.url.startsWith("http")) {
img.src = this.url;
} else {
2022-09-12 20:50:37 +08:00
img.src = `fs/${this.url}`;
2022-09-06 20:36:06 +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";
2022-08-23 14:12:24 +08:00
return img;
}
}
export function inlineImagesPlugin() {
return decoratorStateField((state: EditorState) => {
const widgets: Range<Decoration>[] = [];
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
2022-08-23 14:12:24 +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(
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
2022-08-23 14:12:24 +08:00
const url = imageRexexResult.groups.url;
const title = imageRexexResult.groups.title;
widgets.push(
Decoration.widget({
widget: new InlineImageWidget(url, title),
}).range(node.to),
);
2022-08-29 21:47:16 +08:00
},
2022-08-23 14:12:24 +08:00
});
return Decoration.set(widgets, true);
});
}