2024-05-14 08:45:41 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-10-01 23:07:28 +02:00
|
|
|
import { getAllGlobalKeyBindings } from "@/app/store/keymodel";
|
2024-09-12 03:03:55 +02:00
|
|
|
import { waveEventSubscribe } from "@/app/store/wps";
|
2024-09-16 20:59:39 +02:00
|
|
|
import { RpcApi } from "@/app/store/wshclientapi";
|
2024-10-17 23:50:36 +02:00
|
|
|
import { makeFeBlockRouteId } from "@/app/store/wshrouter";
|
|
|
|
import { DefaultRouter, TabRpcClient } from "@/app/store/wshrpcutil";
|
|
|
|
import { TermWshClient } from "@/app/view/term/term-wsh";
|
2024-07-25 05:34:22 +02:00
|
|
|
import { VDomView } from "@/app/view/term/vdom";
|
2024-10-17 23:50:36 +02:00
|
|
|
import { VDomModel } from "@/app/view/term/vdom-model";
|
|
|
|
import { NodeModel } from "@/layout/index";
|
2024-10-07 23:08:57 +02:00
|
|
|
import { WOS, atoms, getConnStatusAtom, getSettingsKeyAtom, globalStore, useSettingsPrefixAtom } from "@/store/global";
|
2024-06-12 02:42:10 +02:00
|
|
|
import * as services from "@/store/services";
|
2024-06-24 23:34:31 +02:00
|
|
|
import * as keyutil from "@/util/keyutil";
|
2024-06-13 23:41:28 +02:00
|
|
|
import clsx from "clsx";
|
2024-06-14 08:54:04 +02:00
|
|
|
import * as jotai from "jotai";
|
2024-05-28 21:12:28 +02:00
|
|
|
import * as React from "react";
|
2024-06-18 07:38:48 +02:00
|
|
|
import { TermStickers } from "./termsticker";
|
2024-07-25 05:34:22 +02:00
|
|
|
import { TermThemeUpdater } from "./termtheme";
|
2024-07-31 04:52:50 +02:00
|
|
|
import { computeTheme } from "./termutil";
|
2024-06-18 07:38:48 +02:00
|
|
|
import { TermWrap } from "./termwrap";
|
2024-09-04 03:11:28 +02:00
|
|
|
import "./xterm.css";
|
2024-05-14 08:45:41 +02:00
|
|
|
|
2024-05-29 09:28:25 +02:00
|
|
|
type InitialLoadDataType = {
|
|
|
|
loaded: boolean;
|
|
|
|
heldData: Uint8Array[];
|
|
|
|
};
|
|
|
|
|
2024-07-23 02:08:18 +02:00
|
|
|
class TermViewModel {
|
2024-08-23 01:25:53 +02:00
|
|
|
viewType: string;
|
2024-10-17 23:50:36 +02:00
|
|
|
nodeModel: NodeModel;
|
2024-08-24 03:12:40 +02:00
|
|
|
connected: boolean;
|
2024-07-23 02:08:18 +02:00
|
|
|
termRef: React.RefObject<TermWrap>;
|
|
|
|
blockAtom: jotai.Atom<Block>;
|
|
|
|
termMode: jotai.Atom<string>;
|
|
|
|
blockId: string;
|
2024-07-26 01:45:07 +02:00
|
|
|
viewIcon: jotai.Atom<string>;
|
2024-07-26 03:05:32 +02:00
|
|
|
viewName: jotai.Atom<string>;
|
2024-07-31 04:52:50 +02:00
|
|
|
blockBg: jotai.Atom<MetaType>;
|
2024-08-27 01:19:03 +02:00
|
|
|
manageConnection: jotai.Atom<boolean>;
|
2024-09-05 09:21:08 +02:00
|
|
|
connStatus: jotai.Atom<ConnStatus>;
|
2024-10-17 23:50:36 +02:00
|
|
|
termWshClient: TermWshClient;
|
|
|
|
shellProcStatusRef: React.MutableRefObject<string>;
|
|
|
|
vdomModel: VDomModel;
|
2024-07-23 02:08:18 +02:00
|
|
|
|
2024-10-17 23:50:36 +02:00
|
|
|
constructor(blockId: string, nodeModel: NodeModel) {
|
2024-08-23 01:25:53 +02:00
|
|
|
this.viewType = "term";
|
2024-07-23 02:08:18 +02:00
|
|
|
this.blockId = blockId;
|
2024-10-17 23:50:36 +02:00
|
|
|
this.termWshClient = new TermWshClient(blockId, this);
|
|
|
|
DefaultRouter.registerRoute(makeFeBlockRouteId(blockId), this.termWshClient);
|
|
|
|
this.nodeModel = nodeModel;
|
|
|
|
this.vdomModel = new VDomModel(blockId, nodeModel, null, this.termWshClient);
|
2024-07-23 02:08:18 +02:00
|
|
|
this.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
|
|
|
|
this.termMode = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
return blockData?.meta?.["term:mode"] ?? "term";
|
|
|
|
});
|
2024-07-26 03:05:32 +02:00
|
|
|
this.viewIcon = jotai.atom((get) => {
|
|
|
|
return "terminal";
|
|
|
|
});
|
|
|
|
this.viewName = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
2024-07-30 21:33:28 +02:00
|
|
|
if (blockData?.meta?.controller == "cmd") {
|
2024-07-26 03:05:32 +02:00
|
|
|
return "Command";
|
|
|
|
}
|
|
|
|
return "Terminal";
|
|
|
|
});
|
2024-08-27 01:19:03 +02:00
|
|
|
this.manageConnection = jotai.atom(true);
|
2024-07-31 04:52:50 +02:00
|
|
|
this.blockBg = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
2024-08-28 03:49:49 +02:00
|
|
|
const fullConfig = get(atoms.fullConfigAtom);
|
2024-10-07 23:08:57 +02:00
|
|
|
let themeName: string = globalStore.get(getSettingsKeyAtom("term:theme"));
|
|
|
|
if (blockData?.meta?.["term:theme"]) {
|
|
|
|
themeName = blockData.meta["term:theme"];
|
|
|
|
}
|
|
|
|
const theme = computeTheme(fullConfig, themeName);
|
2024-07-31 04:52:50 +02:00
|
|
|
if (theme != null && theme.background != null) {
|
|
|
|
return { bg: theme.background };
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
2024-09-05 09:21:08 +02:00
|
|
|
this.connStatus = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const connName = blockData?.meta?.connection;
|
|
|
|
const connAtom = getConnStatusAtom(connName);
|
|
|
|
return get(connAtom);
|
|
|
|
});
|
2024-08-24 03:12:40 +02:00
|
|
|
}
|
|
|
|
|
2024-10-17 23:50:36 +02:00
|
|
|
dispose() {
|
|
|
|
DefaultRouter.unregisterRoute(makeFeBlockRouteId(this.blockId));
|
|
|
|
}
|
|
|
|
|
2024-07-23 02:08:18 +02:00
|
|
|
giveFocus(): boolean {
|
|
|
|
let termMode = globalStore.get(this.termMode);
|
|
|
|
if (termMode == "term") {
|
|
|
|
if (this.termRef?.current?.terminal) {
|
|
|
|
this.termRef.current.terminal.focus();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2024-07-31 04:52:50 +02:00
|
|
|
|
2024-10-17 23:50:36 +02:00
|
|
|
keyDownHandler(waveEvent: WaveKeyboardEvent): boolean {
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Cmd:Escape")) {
|
|
|
|
const blockAtom = WOS.getWaveObjectAtom<Block>(`block:${this.blockId}`);
|
|
|
|
const blockData = globalStore.get(blockAtom);
|
|
|
|
const newTermMode = blockData?.meta?.["term:mode"] == "html" ? null : "html";
|
|
|
|
RpcApi.SetMetaCommand(TabRpcClient, {
|
|
|
|
oref: WOS.makeORef("block", this.blockId),
|
|
|
|
meta: { "term:mode": newTermMode },
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const blockData = globalStore.get(this.blockAtom);
|
|
|
|
if (blockData.meta?.["term:mode"] == "html") {
|
|
|
|
return this.vdomModel?.globalKeydownHandler(waveEvent);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
handleTerminalKeydown(event: KeyboardEvent): boolean {
|
|
|
|
const waveEvent = keyutil.adaptFromReactOrNativeKeyEvent(event);
|
|
|
|
if (waveEvent.type != "keydown") {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (this.keyDownHandler(waveEvent)) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// deal with terminal specific keybindings
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Ctrl:Shift:v")) {
|
|
|
|
const p = navigator.clipboard.readText();
|
|
|
|
p.then((text) => {
|
|
|
|
this.termRef.current?.terminal.paste(text);
|
|
|
|
});
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return false;
|
|
|
|
} else if (keyutil.checkKeyPressed(waveEvent, "Ctrl:Shift:c")) {
|
|
|
|
const sel = this.termRef.current?.terminal.getSelection();
|
|
|
|
navigator.clipboard.writeText(sel);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (this.shellProcStatusRef.current != "running" && keyutil.checkKeyPressed(waveEvent, "Enter")) {
|
|
|
|
// restart
|
|
|
|
const tabId = globalStore.get(atoms.staticTabId);
|
|
|
|
const prtn = RpcApi.ControllerResyncCommand(TabRpcClient, { tabid: tabId, blockid: this.blockId });
|
|
|
|
prtn.catch((e) => console.log("error controller resync (enter)", this.blockId, e));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const globalKeys = getAllGlobalKeyBindings();
|
|
|
|
for (const key of globalKeys) {
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, key)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-07-31 04:52:50 +02:00
|
|
|
setTerminalTheme(themeName: string) {
|
2024-10-17 23:34:02 +02:00
|
|
|
RpcApi.SetMetaCommand(TabRpcClient, {
|
2024-09-16 20:59:39 +02:00
|
|
|
oref: WOS.makeORef("block", this.blockId),
|
|
|
|
meta: { "term:theme": themeName },
|
|
|
|
});
|
2024-07-31 04:52:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
getSettingsMenuItems(): ContextMenuItem[] {
|
2024-08-30 06:35:22 +02:00
|
|
|
const fullConfig = globalStore.get(atoms.fullConfigAtom);
|
|
|
|
const termThemes = fullConfig?.termthemes ?? {};
|
|
|
|
const termThemeKeys = Object.keys(termThemes);
|
2024-09-05 09:21:08 +02:00
|
|
|
|
2024-08-30 06:35:22 +02:00
|
|
|
termThemeKeys.sort((a, b) => {
|
|
|
|
return termThemes[a]["display:order"] - termThemes[b]["display:order"];
|
|
|
|
});
|
2024-09-05 09:21:08 +02:00
|
|
|
const fullMenu: ContextMenuItem[] = [];
|
2024-08-30 06:35:22 +02:00
|
|
|
const submenu: ContextMenuItem[] = termThemeKeys.map((themeName) => {
|
|
|
|
return {
|
|
|
|
label: termThemes[themeName]["display:name"] ?? themeName,
|
|
|
|
click: () => this.setTerminalTheme(themeName),
|
|
|
|
};
|
|
|
|
});
|
2024-09-05 09:21:08 +02:00
|
|
|
fullMenu.push({
|
|
|
|
label: "Themes",
|
|
|
|
submenu: submenu,
|
|
|
|
});
|
|
|
|
fullMenu.push({ type: "separator" });
|
|
|
|
fullMenu.push({
|
|
|
|
label: "Force Restart Controller",
|
|
|
|
click: () => {
|
|
|
|
const termsize = {
|
|
|
|
rows: this.termRef.current?.terminal?.rows,
|
|
|
|
cols: this.termRef.current?.terminal?.cols,
|
|
|
|
};
|
2024-10-17 23:34:02 +02:00
|
|
|
const prtn = RpcApi.ControllerResyncCommand(TabRpcClient, {
|
|
|
|
tabid: globalStore.get(atoms.staticTabId),
|
2024-09-05 09:21:08 +02:00
|
|
|
blockid: this.blockId,
|
|
|
|
forcerestart: true,
|
|
|
|
rtopts: { termsize: termsize },
|
|
|
|
});
|
|
|
|
prtn.catch((e) => console.log("error controller resync (force restart)", e));
|
2024-07-31 04:52:50 +02:00
|
|
|
},
|
2024-09-05 09:21:08 +02:00
|
|
|
});
|
|
|
|
return fullMenu;
|
2024-07-31 04:52:50 +02:00
|
|
|
}
|
2024-07-23 02:08:18 +02:00
|
|
|
}
|
|
|
|
|
2024-10-17 23:50:36 +02:00
|
|
|
function makeTerminalModel(blockId: string, nodeModel: NodeModel): TermViewModel {
|
|
|
|
return new TermViewModel(blockId, nodeModel);
|
2024-07-23 02:08:18 +02:00
|
|
|
}
|
|
|
|
|
2024-07-25 05:34:22 +02:00
|
|
|
interface TerminalViewProps {
|
|
|
|
blockId: string;
|
|
|
|
model: TermViewModel;
|
|
|
|
}
|
|
|
|
|
2024-09-05 09:21:08 +02:00
|
|
|
const TermResyncHandler = React.memo(({ blockId, model }: TerminalViewProps) => {
|
|
|
|
const connStatus = jotai.useAtomValue(model.connStatus);
|
|
|
|
const [lastConnStatus, setLastConnStatus] = React.useState<ConnStatus>(connStatus);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
if (!model.termRef.current?.hasResized) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const isConnected = connStatus?.status == "connected";
|
|
|
|
const wasConnected = lastConnStatus?.status == "connected";
|
|
|
|
const curConnName = connStatus?.connection;
|
|
|
|
const lastConnName = lastConnStatus?.connection;
|
|
|
|
if (isConnected == wasConnected && curConnName == lastConnName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
model.termRef.current?.resyncController("resync handler");
|
|
|
|
setLastConnStatus(connStatus);
|
|
|
|
}, [connStatus]);
|
|
|
|
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
|
2024-07-25 05:34:22 +02:00
|
|
|
const TerminalView = ({ blockId, model }: TerminalViewProps) => {
|
2024-10-17 23:50:36 +02:00
|
|
|
const viewRef = React.useRef<HTMLDivElement>(null);
|
2024-05-14 08:45:41 +02:00
|
|
|
const connectElemRef = React.useRef<HTMLDivElement>(null);
|
2024-06-18 07:38:48 +02:00
|
|
|
const termRef = React.useRef<TermWrap>(null);
|
2024-07-23 02:08:18 +02:00
|
|
|
model.termRef = termRef;
|
2024-10-17 23:50:36 +02:00
|
|
|
const spstatusRef = React.useRef<string>(null);
|
|
|
|
model.shellProcStatusRef = spstatusRef;
|
2024-06-13 23:41:28 +02:00
|
|
|
const [blockData] = WOS.useWaveObjectValue<Block>(WOS.makeORef("block", blockId));
|
2024-08-28 03:49:49 +02:00
|
|
|
const termSettingsAtom = useSettingsPrefixAtom("term");
|
2024-06-21 22:23:07 +02:00
|
|
|
const termSettings = jotai.useAtomValue(termSettingsAtom);
|
2024-10-17 23:50:36 +02:00
|
|
|
let termMode = blockData?.meta?.["term:mode"] ?? "term";
|
|
|
|
if (termMode != "term" && termMode != "html") {
|
|
|
|
termMode = "term";
|
|
|
|
}
|
|
|
|
const termModeRef = React.useRef(termMode);
|
2024-07-25 05:34:22 +02:00
|
|
|
|
2024-05-14 08:45:41 +02:00
|
|
|
React.useEffect(() => {
|
2024-08-28 03:49:49 +02:00
|
|
|
const fullConfig = globalStore.get(atoms.fullConfigAtom);
|
|
|
|
const termTheme = computeTheme(fullConfig, blockData?.meta?.["term:theme"]);
|
2024-07-31 04:52:50 +02:00
|
|
|
const themeCopy = { ...termTheme };
|
|
|
|
themeCopy.background = "#00000000";
|
2024-10-07 18:51:23 +02:00
|
|
|
let termScrollback = 1000;
|
|
|
|
if (termSettings?.["term:scrollback"]) {
|
|
|
|
termScrollback = Math.floor(termSettings["term:scrollback"]);
|
|
|
|
}
|
|
|
|
if (blockData?.meta?.["term:scrollback"]) {
|
|
|
|
termScrollback = Math.floor(blockData.meta["term:scrollback"]);
|
|
|
|
}
|
|
|
|
if (termScrollback < 0) {
|
|
|
|
termScrollback = 0;
|
|
|
|
}
|
|
|
|
if (termScrollback > 10000) {
|
|
|
|
termScrollback = 10000;
|
|
|
|
}
|
2024-06-24 23:34:31 +02:00
|
|
|
const termWrap = new TermWrap(
|
|
|
|
blockId,
|
|
|
|
connectElemRef.current,
|
|
|
|
{
|
2024-07-31 04:52:50 +02:00
|
|
|
theme: themeCopy,
|
2024-08-28 03:49:49 +02:00
|
|
|
fontSize: termSettings?.["term:fontsize"] ?? 12,
|
|
|
|
fontFamily: termSettings?.["term:fontfamily"] ?? "Hack",
|
2024-06-24 23:34:31 +02:00
|
|
|
drawBoldTextInBrightColors: false,
|
|
|
|
fontWeight: "normal",
|
|
|
|
fontWeightBold: "bold",
|
2024-08-24 07:12:27 +02:00
|
|
|
allowTransparency: true,
|
2024-10-07 18:51:23 +02:00
|
|
|
scrollback: termScrollback,
|
2024-06-24 23:34:31 +02:00
|
|
|
},
|
|
|
|
{
|
2024-10-17 23:50:36 +02:00
|
|
|
keydownHandler: model.handleTerminalKeydown.bind(model),
|
2024-08-28 03:49:49 +02:00
|
|
|
useWebGl: !termSettings?.["term:disablewebgl"],
|
2024-06-24 23:34:31 +02:00
|
|
|
}
|
|
|
|
);
|
2024-06-18 07:38:48 +02:00
|
|
|
(window as any).term = termWrap;
|
|
|
|
termRef.current = termWrap;
|
2024-06-12 02:42:10 +02:00
|
|
|
const rszObs = new ResizeObserver(() => {
|
2024-06-18 07:38:48 +02:00
|
|
|
termWrap.handleResize_debounced();
|
2024-06-12 02:42:10 +02:00
|
|
|
});
|
|
|
|
rszObs.observe(connectElemRef.current);
|
2024-06-18 07:38:48 +02:00
|
|
|
termWrap.initTerminal();
|
2024-06-07 00:08:39 +02:00
|
|
|
return () => {
|
2024-06-18 07:38:48 +02:00
|
|
|
termWrap.dispose();
|
2024-06-29 02:53:35 +02:00
|
|
|
rszObs.disconnect();
|
2024-06-07 00:08:39 +02:00
|
|
|
};
|
2024-07-25 05:34:22 +02:00
|
|
|
}, [blockId, termSettings]);
|
2024-06-06 23:57:37 +02:00
|
|
|
|
2024-10-17 23:50:36 +02:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (termModeRef.current == "html" && termMode == "term") {
|
|
|
|
// focus the terminal
|
|
|
|
model.giveFocus();
|
2024-06-14 08:54:04 +02:00
|
|
|
}
|
2024-10-17 23:50:36 +02:00
|
|
|
termModeRef.current = termMode;
|
|
|
|
}, [termMode]);
|
2024-06-14 08:54:04 +02:00
|
|
|
|
2024-07-03 23:31:02 +02:00
|
|
|
// set intitial controller status, and then subscribe for updates
|
2024-06-24 23:34:31 +02:00
|
|
|
React.useEffect(() => {
|
|
|
|
function updateShellProcStatus(status: string) {
|
|
|
|
if (status == null) {
|
|
|
|
return;
|
|
|
|
}
|
2024-10-17 23:50:36 +02:00
|
|
|
model.shellProcStatusRef.current = status;
|
2024-06-24 23:34:31 +02:00
|
|
|
if (status == "running") {
|
|
|
|
termRef.current?.setIsRunning(true);
|
|
|
|
} else {
|
|
|
|
termRef.current?.setIsRunning(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const initialRTStatus = services.BlockService.GetControllerStatus(blockId);
|
|
|
|
initialRTStatus.then((rts) => {
|
|
|
|
updateShellProcStatus(rts?.shellprocstatus);
|
|
|
|
});
|
2024-09-12 03:03:55 +02:00
|
|
|
return waveEventSubscribe({
|
|
|
|
eventType: "controllerstatus",
|
|
|
|
scope: WOS.makeORef("block", blockId),
|
|
|
|
handler: (event) => {
|
|
|
|
console.log("term waveEvent handler", event);
|
|
|
|
let bcRTS: BlockControllerRuntimeStatus = event.data;
|
|
|
|
updateShellProcStatus(bcRTS?.shellprocstatus);
|
|
|
|
},
|
2024-06-24 23:34:31 +02:00
|
|
|
});
|
2024-07-03 23:31:02 +02:00
|
|
|
}, []);
|
2024-06-24 23:34:31 +02:00
|
|
|
|
2024-06-18 07:38:48 +02:00
|
|
|
let stickerConfig = {
|
|
|
|
charWidth: 8,
|
|
|
|
charHeight: 16,
|
|
|
|
rows: termRef.current?.terminal.rows ?? 24,
|
|
|
|
cols: termRef.current?.terminal.cols ?? 80,
|
|
|
|
blockId: blockId,
|
|
|
|
};
|
|
|
|
|
2024-05-15 01:53:03 +02:00
|
|
|
return (
|
2024-09-03 09:02:03 +02:00
|
|
|
<div className={clsx("view-term", "term-mode-" + termMode)} ref={viewRef}>
|
2024-09-05 09:21:08 +02:00
|
|
|
<TermResyncHandler blockId={blockId} model={model} />
|
2024-07-25 05:34:22 +02:00
|
|
|
<TermThemeUpdater blockId={blockId} termRef={termRef} />
|
2024-06-18 07:38:48 +02:00
|
|
|
<TermStickers config={stickerConfig} />
|
2024-05-15 01:53:03 +02:00
|
|
|
<div key="conntectElem" className="term-connectelem" ref={connectElemRef}></div>
|
2024-10-17 23:50:36 +02:00
|
|
|
<div key="htmlElem" className="term-htmlelem">
|
2024-06-13 23:41:28 +02:00
|
|
|
<div key="htmlElemContent" className="term-htmlelem-content">
|
2024-10-17 23:50:36 +02:00
|
|
|
<VDomView blockId={blockId} nodeModel={model.nodeModel} viewRef={viewRef} model={model.vdomModel} />
|
2024-06-13 23:41:28 +02:00
|
|
|
</div>
|
|
|
|
</div>
|
2024-05-15 01:53:03 +02:00
|
|
|
</div>
|
|
|
|
);
|
2024-05-14 08:45:41 +02:00
|
|
|
};
|
|
|
|
|
2024-08-22 00:49:23 +02:00
|
|
|
export { TermViewModel, TerminalView, makeTerminalModel };
|