2022-08-23 14:12:24 +08:00
|
|
|
import { syntaxTree } from "@codemirror/language";
|
|
|
|
import { Range } from "@codemirror/state";
|
2022-08-29 21:47:16 +08:00
|
|
|
import {
|
|
|
|
Decoration,
|
|
|
|
DecorationSet,
|
|
|
|
EditorView,
|
|
|
|
ViewPlugin,
|
|
|
|
ViewUpdate,
|
|
|
|
WidgetType,
|
|
|
|
} from "@codemirror/view";
|
2022-08-23 14:12:24 +08:00
|
|
|
|
|
|
|
class InlineImageWidget extends WidgetType {
|
2022-08-24 00:53:05 +08:00
|
|
|
constructor(readonly url: string, readonly title: string) {
|
2022-08-23 14:12:24 +08:00
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
toDOM() {
|
2022-08-29 21:47:16 +08:00
|
|
|
const img = document.createElement("img");
|
2022-08-23 14:12:24 +08:00
|
|
|
img.src = this.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";
|
2022-08-23 14:12:24 +08:00
|
|
|
|
|
|
|
return img;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const inlineImages = (view: EditorView) => {
|
|
|
|
let widgets: Range<Decoration>[] = [];
|
2022-08-24 00:53:05 +08:00
|
|
|
const imageRegex = /!\[(?<title>[^\]]*)\]\((?<url>.+)\)/;
|
2022-08-23 14:12:24 +08:00
|
|
|
|
2022-08-29 21:47:16 +08:00
|
|
|
for (let { from, to } of view.visibleRanges) {
|
2022-08-23 14:12:24 +08:00
|
|
|
syntaxTree(view.state).iterate({
|
2022-08-29 21:47:16 +08:00
|
|
|
from,
|
|
|
|
to,
|
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(
|
|
|
|
view.state.sliceDoc(node.from, node.to)
|
|
|
|
);
|
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;
|
2022-08-24 00:53:05 +08:00
|
|
|
const title = imageRexexResult.groups.title;
|
2022-08-23 14:12:24 +08:00
|
|
|
let deco = Decoration.widget({
|
2022-08-24 00:53:05 +08:00
|
|
|
widget: new InlineImageWidget(url, title),
|
2022-08-23 14:12:24 +08:00
|
|
|
});
|
|
|
|
widgets.push(deco.range(node.to));
|
2022-08-29 21:47:16 +08:00
|
|
|
},
|
2022-08-23 14:12:24 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return Decoration.set(widgets, true);
|
2022-08-29 21:47:16 +08:00
|
|
|
};
|
2022-08-23 14:12:24 +08: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,
|
|
|
|
}
|
|
|
|
);
|