import { WidgetType } from "@codemirror/view"; import type { Client } from "../client.ts"; import type { CodeWidgetButton, CodeWidgetCallback, } from "../../plug-api/types.ts"; import { renderMarkdownToHtml } from "../../plugs/markdown/markdown_render.ts"; import { isLocalPath, resolvePath, } from "@silverbulletmd/silverbullet/lib/resolve"; import { parse } from "$common/markdown_parser/parse_tree.ts"; import { extendedMarkdownLanguage } from "$common/markdown_parser/parser.ts"; import { renderToText } from "@silverbulletmd/silverbullet/lib/tree"; import { attachWidgetEventHandlers } from "./widget_util.ts"; export const activeWidgets = new Set(); export interface DomWidget { dom?: HTMLElement; renderContent( div: HTMLElement, cachedHtml: string | undefined, ): Promise; } export class MarkdownWidget extends WidgetType { public dom?: HTMLElement; constructor( readonly from: number | undefined, readonly client: Client, readonly cacheKey: string, readonly bodyText: string, readonly codeWidgetCallback: CodeWidgetCallback, readonly className: string, private tryInline = false, ) { super(); } toDOM(): HTMLElement { const div = document.createElement("div"); div.className = this.className; const cacheItem = this.client.getWidgetCache(this.cacheKey); if (cacheItem) { div.innerHTML = this.wrapHtml(cacheItem.html, cacheItem.buttons); if (cacheItem.html) { this.attachListeners(div, cacheItem.buttons); } } // Async kick-off of content renderer this.renderContent(div, cacheItem?.html).catch(console.error); this.dom = div; return div; } async renderContent( div: HTMLElement, cachedHtml: string | undefined, ) { const widgetContent = await this.codeWidgetCallback( this.bodyText, this.client.currentPage, ); activeWidgets.add(this); if (!widgetContent) { div.innerHTML = ""; this.client.setWidgetCache( this.cacheKey, { height: div.clientHeight, html: "" }, ); return; } let mdTree = parse( extendedMarkdownLanguage, widgetContent.markdown!, ); mdTree = await this.client.clientSystem.localSyscall( "system.invokeFunction", [ "markdown.expandCodeWidgets", mdTree, this.client.currentPage, ], ); const trimmedMarkdown = renderToText(mdTree).trim(); if (!trimmedMarkdown) { // Net empty result after expansion div.innerHTML = ""; this.client.setWidgetCache( this.cacheKey, { height: div.clientHeight, html: "" }, ); return; } if (this.tryInline) { if (trimmedMarkdown.includes("\n")) { // Heuristic that this is going to be a multi-line output and we should render this as a HTML block div.style.display = "block"; } else { div.style.display = "inline"; } } // Parse the markdown again after trimming mdTree = parse( extendedMarkdownLanguage, trimmedMarkdown, ); const html = renderMarkdownToHtml(mdTree, { // Annotate every element with its position so we can use it to put // the cursor there when the user clicks on the table. annotationPositions: true, translateUrls: (url) => { if (isLocalPath(url)) { url = resolvePath( this.client.currentPage, decodeURI(url), ); } return url; }, preserveAttributes: true, }, this.client.ui.viewState.allPages); if (cachedHtml === html) { // HTML still same as in cache, no need to re-render return; } div.innerHTML = this.wrapHtml( html, widgetContent.buttons, ); if (html) { this.attachListeners(div, widgetContent.buttons); } // Let's give it a tick, then measure and cache setTimeout(() => { this.client.setWidgetCache( this.cacheKey, { height: div.offsetHeight, html, buttons: widgetContent.buttons, }, ); // Because of the rejiggering of the DOM, we need to do a no-op cursor move to make sure it's positioned correctly this.client.editorView.dispatch({ selection: { anchor: this.client.editorView.state.selection.main.anchor, }, }); }); } private wrapHtml( html: string, buttons: CodeWidgetButton[] = [], ) { if (!html) { return ""; } if (buttons.length === 0) { return html; } else { return `
${ buttons.filter((button) => !button.widgetTarget).map((button, idx) => ` ` ).join("") }
${html}
`; } } private attachListeners(div: HTMLElement, buttons?: CodeWidgetButton[]) { attachWidgetEventHandlers(div, this.client); if (!buttons) { buttons = []; } for (let i = 0; i < buttons.length; i++) { const button = buttons[i]; if (button.widgetTarget) { div.addEventListener("click", () => { console.log("Widget clicked"); this.client.clientSystem.localSyscall("system.invokeFunction", [ button.invokeFunction[0], this.from, ]).catch(console.error); }); } else { div.querySelector(`button[data-button="${i}"]`)!.addEventListener( "click", (e) => { e.stopPropagation(); this.client.clientSystem.localSyscall( "system.invokeFunction", button.invokeFunction, ).then((newContent: string | undefined) => { if (newContent) { div.innerText = newContent; } this.client.focus(); }).catch(console.error); }, ); } } } override get estimatedHeight(): number { const cacheItem = this.client.getWidgetCache(this.cacheKey); return cacheItem ? cacheItem.height : -1; } override eq(other: WidgetType): boolean { return ( other instanceof MarkdownWidget && other.bodyText === this.bodyText && other.cacheKey === this.cacheKey ); } } export function reloadAllWidgets() { for (const widget of activeWidgets) { // Garbage collect as we go if (!widget.dom || !widget.dom.parentNode) { activeWidgets.delete(widget); continue; } widget.renderContent(widget.dom!, undefined).catch(console.error); } } function garbageCollectWidgets() { for (const widget of activeWidgets) { if (!widget.dom || !widget.dom.parentNode) { // console.log("Garbage collecting widget", widget.bodyText); activeWidgets.delete(widget); } } } setInterval(garbageCollectWidgets, 5000);