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 "$sb/lib/resolve.ts"; import { parse } from "$common/markdown_parser/parse_tree.ts"; import { parsePageRef } from "../../plug-api/lib/page_ref.ts"; import { extendedMarkdownLanguage } from "$common/markdown_parser/parser.ts"; import { tagPrefix } from "../../plugs/index/constants.ts"; import { renderToText } from "$sb/lib/tree.ts"; const activeWidgets = new Set(); 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, ) { 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; } // 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 ""; } return `
${ buttons.filter((button) => !button.widgetTarget).map((button, idx) => ` ` ).join("") }
${html}
`; } private attachListeners(div: HTMLElement, buttons?: CodeWidgetButton[]) { div.addEventListener("mousedown", (e) => { // CodeMirror overrides mousedown on parent elements to implement its own selection highlighting. // That's nice, but not for markdown widgets, so let's not propagate the event to CodeMirror here. e.stopPropagation(); }); // Override wiki links with local navigate (faster) div.querySelectorAll("a[data-ref]").forEach((el_) => { const el = el_ as HTMLElement; // Override default click behavior with a local navigate (faster) el.addEventListener("click", (e) => { if (e.ctrlKey || e.metaKey) { // Don't do anything special for ctrl/meta clicks return; } e.preventDefault(); e.stopPropagation(); const pageRef = parsePageRef(el.dataset.ref!); this.client.navigate(pageRef, false, e.ctrlKey || e.metaKey); }); }); // Attach click handlers to hash tags div.querySelectorAll("span.hashtag").forEach((el_) => { const el = el_ as HTMLElement; // Override default click behavior with a local navigate (faster) el.addEventListener("click", (e) => { if (e.ctrlKey || e.metaKey) { // Don't do anything special for ctrl/meta clicks return; } this.client.navigate({ page: `${tagPrefix}${el.innerText.slice(1)}`, pos: 0, }); }); }); div.querySelectorAll("button[data-onclick]").forEach((el_) => { const el = el_ as HTMLElement; const onclick = el.dataset.onclick!; const parsedOnclick = JSON.parse(onclick); if (parsedOnclick[0] === "command") { const command = parsedOnclick[1]; el.addEventListener("click", (e) => { e.preventDefault(); e.stopPropagation(); console.info( "Command link clicked in widget, running", parsedOnclick, ); this.client.runCommandByName(command, parsedOnclick[2]).catch( console.error, ); }); } }); // Implement task toggling div.querySelectorAll("span[data-external-task-ref]").forEach((el: any) => { const taskRef = el.dataset.externalTaskRef; const input = el.querySelector("input[type=checkbox]")!; input.addEventListener( "click", (e: any) => { // Avoid triggering the click on the parent e.stopPropagation(); }, ); input.addEventListener( "change", (e: any) => { e.stopPropagation(); const oldState = e.target.dataset.state; const newState = oldState === " " ? "x" : " "; // Update state in DOM as well for future toggles e.target.dataset.state = newState; console.log("Toggling task", taskRef); this.client.clientSystem.localSyscall( "system.invokeFunction", ["tasks.updateTaskState", taskRef, oldState, newState], ).catch( console.error, ); }, ); }); 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, 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, this.bodyText, ]).then((newContent: string | undefined) => { if (newContent) { div.innerText = newContent; } this.client.focus(); }).catch(console.error); }, ); } } } get estimatedHeight(): number { const cacheItem = this.client.getWidgetCache(this.cacheKey); return cacheItem ? cacheItem.height : -1; } eq(other: WidgetType): boolean { return ( other instanceof MarkdownWidget && other.bodyText === this.bodyText && other.cacheKey === this.cacheKey ); } } export function reloadAllMarkdownWidgets() { 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);