// Copyright 2024, Command Line Inc. // SPDX-License-Identifier: Apache-2.0 import { Markdown } from "@/app/element/markdown"; import { VDomModel } from "@/app/view/vdom/vdom-model"; import { adaptFromReactOrNativeKeyEvent, checkKeyPressed } from "@/util/keyutil"; import clsx from "clsx"; import debug from "debug"; import * as jotai from "jotai"; import * as React from "react"; import { BlockNodeModel } from "@/app/block/blocktypes"; import { convertVDomId, getTextChildren, validateAndWrapCss } from "@/app/view/vdom/vdom-utils"; import "./vdom.less"; const TextTag = "#text"; const FragmentTag = "#fragment"; const WaveTextTag = "wave:text"; const WaveNullTag = "wave:null"; const StyleTagName = "style"; const VDomObjType_Ref = "ref"; const VDomObjType_Binding = "binding"; const VDomObjType_Func = "func"; const dlog = debug("wave:vdom"); type VDomReactTagType = (props: { elem: VDomElem; model: VDomModel }) => JSX.Element; const WaveTagMap: Record = { "wave:markdown": WaveMarkdown, }; const AllowedSimpleTags: { [tagName: string]: boolean } = { div: true, b: true, i: true, p: true, s: true, span: true, a: true, img: true, h1: true, h2: true, h3: true, h4: true, h5: true, h6: true, ul: true, ol: true, li: true, input: true, button: true, textarea: true, select: true, option: true, form: true, label: true, table: true, thead: true, tbody: true, tr: true, th: true, td: true, hr: true, br: true, pre: true, code: true, }; const IdAttributes = { id: true, for: true, "aria-labelledby": true, "aria-describedby": true, "aria-controls": true, "aria-owns": true, form: true, headers: true, usemap: true, list: true, }; function convertVDomFunc(model: VDomModel, fnDecl: VDomFunc, compId: string, propName: string): (e: any) => void { return (e: any) => { if ((propName == "onKeyDown" || propName == "onKeyDownCapture") && fnDecl["#keys"]) { let waveEvent = adaptFromReactOrNativeKeyEvent(e); for (let keyDesc of fnDecl.keys || []) { if (checkKeyPressed(waveEvent, keyDesc)) { e.preventDefault(); e.stopPropagation(); model.callVDomFunc(fnDecl, e, compId, propName); return; } } return; } if (fnDecl.preventdefault) { e.preventDefault(); } if (fnDecl.stoppropagation) { e.stopPropagation(); } model.callVDomFunc(fnDecl, e, compId, propName); }; } function convertElemToTag(elem: VDomElem, model: VDomModel): JSX.Element | string { if (elem == null) { return null; } if (elem.tag == TextTag) { return elem.text; } return React.createElement(VDomTag, { key: elem.waveid, elem, model }); } function isObject(v: any): boolean { return v != null && !Array.isArray(v) && typeof v === "object"; } function isArray(v: any): boolean { return Array.isArray(v); } function resolveBinding(binding: VDomBinding, model: VDomModel): [any, string[]] { const bindName = binding.bind; if (bindName == null || bindName == "") { return [null, []]; } // for now we only recognize $.[atomname] bindings if (!bindName.startsWith("$.")) { return [null, []]; } const atomName = bindName.substring(2); if (atomName == "") { return [null, []]; } const atom = model.getAtomContainer(atomName); if (atom == null) { return [null, []]; } return [atom.val, [atomName]]; } type GenericPropsType = { [key: string]: any }; // returns props, and a set of atom keys used in the props function convertProps(elem: VDomElem, model: VDomModel): [GenericPropsType, Set] { let props: GenericPropsType = {}; let atomKeys = new Set(); if (elem.props == null) { return [props, atomKeys]; } for (let key in elem.props) { let val = elem.props[key]; if (val == null) { continue; } if (key == "ref") { if (val == null) { continue; } if (isObject(val) && val.type == VDomObjType_Ref) { const valRef = val as VDomRef; const refContainer = model.getOrCreateRefContainer(valRef); props[key] = refContainer.refFn; } continue; } if (isObject(val) && val.type == VDomObjType_Func) { const valFunc = val as VDomFunc; props[key] = convertVDomFunc(model, valFunc, elem.waveid, key); continue; } if (isObject(val) && val.type == VDomObjType_Binding) { const [propVal, atomDeps] = resolveBinding(val as VDomBinding, model); props[key] = propVal; for (let atomDep of atomDeps) { atomKeys.add(atomDep); } continue; } if (key == "style" && isObject(val)) { // assuming the entire style prop wasn't bound, look through the individual keys and bind them for (let styleKey in val) { let styleVal = val[styleKey]; if (isObject(styleVal) && styleVal.type == VDomObjType_Binding) { const [stylePropVal, styleAtomDeps] = resolveBinding(styleVal as VDomBinding, model); val[styleKey] = stylePropVal; for (let styleAtomDep of styleAtomDeps) { atomKeys.add(styleAtomDep); } } } // fallthrough to set props[key] = val } if (IdAttributes[key]) { props[key] = convertVDomId(model, val); continue; } props[key] = val; } return [props, atomKeys]; } function convertChildren(elem: VDomElem, model: VDomModel): (string | JSX.Element)[] { let childrenComps: (string | JSX.Element)[] = []; if (elem.children == null) { return childrenComps; } for (let child of elem.children) { if (child == null) { continue; } childrenComps.push(convertElemToTag(child, model)); } return childrenComps; } function stringSetsEqual(set1: Set, set2: Set): boolean { if (set1.size != set2.size) { return false; } for (let elem of set1) { if (!set2.has(elem)) { return false; } } return true; } function useVDom(model: VDomModel, elem: VDomElem): GenericPropsType { const version = jotai.useAtomValue(model.getVDomNodeVersionAtom(elem)); const [oldAtomKeys, setOldAtomKeys] = React.useState>(new Set()); let [props, atomKeys] = convertProps(elem, model); React.useEffect(() => { if (stringSetsEqual(atomKeys, oldAtomKeys)) { return; } model.tagUnuseAtoms(elem.waveid, oldAtomKeys); model.tagUseAtoms(elem.waveid, atomKeys); setOldAtomKeys(atomKeys); }, [atomKeys]); React.useEffect(() => { return () => { model.tagUnuseAtoms(elem.waveid, oldAtomKeys); }; }, []); return props; } function WaveMarkdown({ elem, model }: { elem: VDomElem; model: VDomModel }) { const props = useVDom(model, elem); return ( ); } function StyleTag({ elem, model }: { elem: VDomElem; model: VDomModel }) { const styleText = getTextChildren(elem); if (styleText == null) { return null; } const wrapperClassName = "vdom-" + model.blockId; // TODO handle errors const sanitizedCss = validateAndWrapCss(model, styleText, wrapperClassName); if (sanitizedCss == null) { return null; } return ; } function VDomTag({ elem, model }: { elem: VDomElem; model: VDomModel }) { const props = useVDom(model, elem); if (elem.tag == WaveNullTag) { return null; } if (elem.tag == WaveTextTag) { return props.text; } const waveTag = WaveTagMap[elem.tag]; if (waveTag) { return waveTag({ elem, model }); } if (elem.tag == StyleTagName) { return ; } if (!AllowedSimpleTags[elem.tag]) { return
{"Invalid Tag <" + elem.tag + ">"}
; } let childrenComps = convertChildren(elem, model); if (elem.tag == FragmentTag) { return childrenComps; } props.key = "e-" + elem.waveid; return React.createElement(elem.tag, props, childrenComps); } function vdomText(text: string): VDomElem { return { tag: "#text", text: text, }; } const testVDom: VDomElem = { waveid: "testid1", tag: "div", children: [ { waveid: "testh1", tag: "h1", children: [vdomText("Hello World")], }, { waveid: "testp", tag: "p", children: [vdomText("This is a paragraph (from VDOM)")], }, ], }; function VDomRoot({ model }: { model: VDomModel }) { let rootNode = jotai.useAtomValue(model.vdomRoot); if (model.viewRef.current == null || rootNode == null) { return null; } dlog("render", rootNode); let rtn = convertElemToTag(rootNode, model); return
{rtn}
; } function makeVDomModel(blockId: string, nodeModel: BlockNodeModel): VDomModel { return new VDomModel(blockId, nodeModel); } type VDomViewProps = { model: VDomModel; blockId: string; }; function VDomView({ blockId, model }: VDomViewProps) { let viewRef = React.useRef(null); model.viewRef = viewRef; const vdomClass = "vdom-" + blockId; return (
); } export { makeVDomModel, VDomView };