2024-08-02 23:14:40 +08:00
|
|
|
import type {
|
|
|
|
CodeWidgetCallback,
|
|
|
|
WidgetContent,
|
|
|
|
} from "../../plug-api/types.ts";
|
2024-03-16 22:29:24 +08:00
|
|
|
import { WidgetType } from "@codemirror/view";
|
2023-12-27 20:38:38 +08:00
|
|
|
import type { Client } from "../client.ts";
|
|
|
|
import { createWidgetSandboxIFrame } from "../components/widget_sandbox_iframe.ts";
|
|
|
|
|
|
|
|
export class IFrameWidget extends WidgetType {
|
|
|
|
iframe?: HTMLIFrameElement;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
readonly from: number,
|
|
|
|
readonly to: number,
|
|
|
|
readonly client: Client,
|
|
|
|
readonly bodyText: string,
|
|
|
|
readonly codeWidgetCallback: CodeWidgetCallback,
|
|
|
|
) {
|
|
|
|
super();
|
|
|
|
}
|
|
|
|
|
|
|
|
toDOM(): HTMLElement {
|
|
|
|
const from = this.from;
|
|
|
|
const iframe = createWidgetSandboxIFrame(
|
|
|
|
this.client,
|
|
|
|
this.bodyText,
|
2024-01-24 21:44:39 +08:00
|
|
|
this.codeWidgetCallback(this.bodyText, this.client.currentPage),
|
2023-12-27 20:38:38 +08:00
|
|
|
(message) => {
|
|
|
|
switch (message.type) {
|
|
|
|
case "blur":
|
|
|
|
this.client.editorView.dispatch({
|
|
|
|
selection: { anchor: from },
|
|
|
|
});
|
|
|
|
this.client.focus();
|
|
|
|
|
|
|
|
break;
|
|
|
|
case "reload":
|
2024-01-24 21:44:39 +08:00
|
|
|
this.codeWidgetCallback(this.bodyText, this.client.currentPage)
|
2023-12-27 20:38:38 +08:00
|
|
|
.then(
|
2024-01-09 00:08:26 +08:00
|
|
|
(widgetContent: WidgetContent | null) => {
|
|
|
|
if (widgetContent === null) {
|
|
|
|
iframe.contentWindow!.postMessage({
|
|
|
|
type: "html",
|
|
|
|
html: "",
|
|
|
|
theme:
|
|
|
|
document.getElementsByTagName("html")[0].dataset.theme,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
iframe.contentWindow!.postMessage({
|
|
|
|
type: "html",
|
|
|
|
html: widgetContent.html,
|
|
|
|
script: widgetContent.script,
|
|
|
|
theme:
|
|
|
|
document.getElementsByTagName("html")[0].dataset.theme,
|
|
|
|
});
|
|
|
|
}
|
2023-12-27 20:38:38 +08:00
|
|
|
},
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
const estimatedHeight = this.estimatedHeight;
|
|
|
|
iframe.height = `${estimatedHeight}px`;
|
|
|
|
|
|
|
|
return iframe;
|
|
|
|
}
|
|
|
|
|
|
|
|
get estimatedHeight(): number {
|
2023-12-28 23:14:30 +08:00
|
|
|
const cachedHeight = this.client.getCachedWidgetHeight(this.bodyText);
|
2024-01-02 19:48:18 +08:00
|
|
|
// console.log("Calling estimated height", this.bodyText, cachedHeight);
|
2023-12-27 20:38:38 +08:00
|
|
|
return cachedHeight > 0 ? cachedHeight : 150;
|
|
|
|
}
|
|
|
|
|
|
|
|
eq(other: WidgetType): boolean {
|
|
|
|
return (
|
|
|
|
other instanceof IFrameWidget &&
|
|
|
|
other.bodyText === this.bodyText
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|