1
0
silverbullet/packages/web/inline_image.ts

85 lines
1.9 KiB
TypeScript
Raw Normal View History

2022-08-23 06:12:24 +00:00
import { syntaxTree } from "@codemirror/language";
import { Range } from "@codemirror/state";
2022-08-29 13:47:16 +00:00
import {
Decoration,
DecorationSet,
EditorView,
ViewPlugin,
ViewUpdate,
WidgetType,
} from "@codemirror/view";
2022-08-23 06:12:24 +00:00
class InlineImageWidget extends WidgetType {
constructor(readonly url: string, readonly title: string) {
2022-08-23 06:12:24 +00:00
super();
}
eq(other: InlineImageWidget) {
return other.url === this.url && other.title === this.title;
2022-08-23 06:12:24 +00:00
}
toDOM() {
2022-08-29 13:47:16 +00:00
const img = document.createElement("img");
2022-08-23 06:12:24 +00:00
img.src = this.url;
img.alt = this.title;
img.title = this.title;
2022-08-29 13:47:16 +00:00
img.style.display = "block";
img.className = "sb-inline-img";
2022-08-23 06:12:24 +00:00
return img;
}
}
const inlineImages = (view: EditorView) => {
let widgets: Range<Decoration>[] = [];
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
2022-08-23 06:12:24 +00:00
2022-08-29 13:47:16 +00:00
for (let { from, to } of view.visibleRanges) {
2022-08-23 06:12:24 +00:00
syntaxTree(view.state).iterate({
2022-08-29 13:47:16 +00:00
from,
to,
2022-08-23 06:12:24 +00:00
enter: (node) => {
2022-08-29 13:47:16 +00:00
if (node.name !== "Image") {
return;
2022-08-23 06:12:24 +00:00
}
2022-08-29 13:47:16 +00:00
const imageRexexResult = imageRegex.exec(
view.state.sliceDoc(node.from, node.to)
);
2022-08-23 06:12:24 +00:00
if (imageRexexResult === null || !imageRexexResult.groups) {
return;
}
2022-08-29 13:47:16 +00:00
2022-08-23 06:12:24 +00:00
const url = imageRexexResult.groups.url;
const title = imageRexexResult.groups.title;
2022-08-23 06:12:24 +00:00
let deco = Decoration.widget({
widget: new InlineImageWidget(url, title),
2022-08-23 06:12:24 +00:00
});
widgets.push(deco.range(node.to));
2022-08-29 13:47:16 +00:00
},
2022-08-23 06:12:24 +00:00
});
}
return Decoration.set(widgets, true);
2022-08-29 13:47:16 +00:00
};
2022-08-23 06:12:24 +00:00
export const inlineImagesPlugin = () =>
ViewPlugin.fromClass(
class {
decorations: DecorationSet;
constructor(view: EditorView) {
this.decorations = inlineImages(view);
}
update(update: ViewUpdate) {
if (update.docChanged) {
this.decorations = inlineImages(update.view);
}
}
},
{
decorations: (v) => v.decorations,
}
);