waveterm/frontend/app/view/term/term.tsx

663 lines
24 KiB
TypeScript
Raw Normal View History

// Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0
2024-10-26 03:36:09 +02:00
import { Block, SubBlock } from "@/app/block/block";
import { BlockNodeModel } from "@/app/block/blocktypes";
import { getAllGlobalKeyBindings } from "@/app/store/keymodel";
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-10-25 22:45:00 +02:00
import { VDomModel } from "@/app/view/vdom/vdom-model";
2024-10-24 07:47:29 +02:00
import {
WOS,
atoms,
getBlockComponentModel,
getConnStatusAtom,
getSettingsKeyAtom,
globalStore,
2024-10-24 20:01:39 +02:00
useBlockAtom,
2024-10-24 07:47:29 +02:00
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";
import clsx from "clsx";
2024-10-24 07:47:29 +02:00
import debug from "debug";
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";
import { TermStickers } from "./termsticker";
import { TermThemeUpdater } from "./termtheme";
import { computeTheme } from "./termutil";
import { TermWrap } from "./termwrap";
import "./xterm.css";
2024-10-24 07:47:29 +02:00
const dlog = debug("wave:term");
type InitialLoadDataType = {
loaded: boolean;
heldData: Uint8Array[];
};
class TermViewModel {
2024-08-23 01:25:53 +02:00
viewType: string;
2024-10-26 03:36:09 +02:00
nodeModel: BlockNodeModel;
connected: boolean;
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>;
viewName: jotai.Atom<string>;
2024-10-24 07:47:29 +02:00
viewText: jotai.Atom<HeaderElem[]>;
blockBg: jotai.Atom<MetaType>;
manageConnection: jotai.Atom<boolean>;
connStatus: jotai.Atom<ConnStatus>;
2024-10-17 23:50:36 +02:00
termWshClient: TermWshClient;
shellProcStatusRef: React.MutableRefObject<string>;
2024-10-24 07:47:29 +02:00
vdomBlockId: jotai.Atom<string>;
2024-11-11 22:11:09 +01:00
vdomToolbarBlockId: jotai.Atom<string>;
vdomToolbarTarget: jotai.PrimitiveAtom<VDomTargetToolbar>;
2024-10-24 20:01:39 +02:00
fontSizeAtom: jotai.Atom<number>;
termThemeNameAtom: jotai.Atom<string>;
2024-11-11 22:11:09 +01:00
noPadding: jotai.PrimitiveAtom<boolean>;
endIconButtons: jotai.Atom<IconButtonDecl[]>;
2024-10-26 03:36:09 +02:00
constructor(blockId: string, nodeModel: BlockNodeModel) {
2024-08-23 01:25:53 +02:00
this.viewType = "term";
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.blockAtom = WOS.getWaveObjectAtom<Block>(`block:${blockId}`);
2024-10-24 07:47:29 +02:00
this.vdomBlockId = jotai.atom((get) => {
const blockData = get(this.blockAtom);
return blockData?.meta?.["term:vdomblockid"];
});
2024-11-11 22:11:09 +01:00
this.vdomToolbarBlockId = jotai.atom((get) => {
const blockData = get(this.blockAtom);
return blockData?.meta?.["term:vdomtoolbarblockid"];
});
this.vdomToolbarTarget = jotai.atom<VDomTargetToolbar>(null) as jotai.PrimitiveAtom<VDomTargetToolbar>;
this.termMode = jotai.atom((get) => {
const blockData = get(this.blockAtom);
return blockData?.meta?.["term:mode"] ?? "term";
});
this.viewIcon = jotai.atom((get) => {
2024-10-24 07:47:29 +02:00
const termMode = get(this.termMode);
if (termMode == "vdom") {
return "bolt";
}
return "terminal";
});
this.viewName = jotai.atom((get) => {
const blockData = get(this.blockAtom);
2024-10-24 07:47:29 +02:00
const termMode = get(this.termMode);
if (termMode == "vdom") {
return "Wave App";
}
if (blockData?.meta?.controller == "cmd") {
return "Command";
}
return "Terminal";
});
2024-10-24 07:47:29 +02:00
this.viewText = jotai.atom((get) => {
const termMode = get(this.termMode);
if (termMode == "vdom") {
return [
{
elemtype: "iconbutton",
icon: "square-terminal",
title: "Switch back to Terminal",
click: () => {
this.setTermMode("term");
},
},
];
} else {
const vdomBlockId = get(this.vdomBlockId);
if (vdomBlockId) {
return [
{
elemtype: "iconbutton",
icon: "bolt",
title: "Switch to Wave App",
click: () => {
this.setTermMode("vdom");
},
},
];
}
}
return null;
});
this.manageConnection = jotai.atom((get) => {
const termMode = get(this.termMode);
if (termMode == "vdom") {
return false;
}
return true;
});
this.blockBg = jotai.atom((get) => {
const blockData = get(this.blockAtom);
2024-08-28 03:49:49 +02:00
const fullConfig = get(atoms.fullConfigAtom);
let themeName: string = get(getSettingsKeyAtom("term:theme"));
if (blockData?.meta?.["term:theme"]) {
themeName = blockData.meta["term:theme"];
}
const theme = computeTheme(fullConfig, themeName);
if (theme != null && theme.background != null) {
return { bg: theme.background };
}
return null;
});
this.connStatus = jotai.atom((get) => {
const blockData = get(this.blockAtom);
const connName = blockData?.meta?.connection;
const connAtom = getConnStatusAtom(connName);
return get(connAtom);
});
2024-10-24 20:01:39 +02:00
this.fontSizeAtom = useBlockAtom(blockId, "fontsizeatom", () => {
return jotai.atom<number>((get) => {
const blockData = get(this.blockAtom);
const fsSettingsAtom = getSettingsKeyAtom("term:fontsize");
const settingsFontSize = get(fsSettingsAtom);
const rtnFontSize = blockData?.meta?.["term:fontsize"] ?? settingsFontSize ?? 12;
if (typeof rtnFontSize != "number" || isNaN(rtnFontSize) || rtnFontSize < 4 || rtnFontSize > 64) {
return 12;
}
return rtnFontSize;
});
});
this.termThemeNameAtom = useBlockAtom(blockId, "termthemeatom", () => {
return jotai.atom<string>((get) => {
const blockData = get(this.blockAtom);
const settingsKeyAtom = getSettingsKeyAtom("term:theme");
return blockData?.meta?.["term:theme"] ?? get(settingsKeyAtom) ?? "default-dark";
});
});
2024-11-11 22:11:09 +01:00
this.noPadding = jotai.atom(true);
this.endIconButtons = jotai.atom((get) => {
const blockData = get(this.blockAtom);
if (blockData?.meta?.["controller"] != "cmd") {
return [];
}
return [
{
elemtype: "iconbutton",
icon: "refresh",
click: this.forceRestartController.bind(this),
title: "Force Restart Controller",
},
];
});
}
2024-10-24 07:47:29 +02:00
setTermMode(mode: "term" | "vdom") {
if (mode == "term") {
mode = null;
}
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", this.blockId),
meta: { "term:mode": mode },
});
}
getVDomModel(): VDomModel {
const vdomBlockId = globalStore.get(this.vdomBlockId);
if (!vdomBlockId) {
return null;
}
const bcm = getBlockComponentModel(vdomBlockId);
if (!bcm) {
return null;
}
return bcm.viewModel as VDomModel;
}
2024-11-11 22:11:09 +01:00
getVDomToolbarModel(): VDomModel {
const vdomToolbarBlockId = globalStore.get(this.vdomToolbarBlockId);
if (!vdomToolbarBlockId) {
return null;
}
const bcm = getBlockComponentModel(vdomToolbarBlockId);
if (!bcm) {
return null;
}
return bcm.viewModel as VDomModel;
}
2024-10-17 23:50:36 +02:00
dispose() {
DefaultRouter.unregisterRoute(makeFeBlockRouteId(this.blockId));
}
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-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);
2024-10-24 07:47:29 +02:00
const newTermMode = blockData?.meta?.["term:mode"] == "vdom" ? null : "vdom";
const vdomBlockId = globalStore.get(this.vdomBlockId);
if (newTermMode == "vdom" && !vdomBlockId) {
return;
}
this.setTermMode(newTermMode);
2024-10-17 23:50:36 +02:00
return true;
}
const blockData = globalStore.get(this.blockAtom);
2024-10-24 07:47:29 +02:00
if (blockData.meta?.["term:mode"] == "vdom") {
const vdomModel = this.getVDomModel();
return vdomModel?.keyDownHandler(waveEvent);
2024-10-17 23:50:36 +02:00
}
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;
}
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 },
});
}
forceRestartController() {
const termsize = {
rows: this.termRef.current?.terminal?.rows,
cols: this.termRef.current?.terminal?.cols,
};
const prtn = RpcApi.ControllerResyncCommand(TabRpcClient, {
tabid: globalStore.get(atoms.staticTabId),
blockid: this.blockId,
forcerestart: true,
rtopts: { termsize: termsize },
});
prtn.catch((e) => console.log("error controller resync (force restart)", e));
}
getSettingsMenuItems(): ContextMenuItem[] {
const fullConfig = globalStore.get(atoms.fullConfigAtom);
const termThemes = fullConfig?.termthemes ?? {};
const termThemeKeys = Object.keys(termThemes);
2024-10-24 20:01:39 +02:00
const curThemeName = globalStore.get(this.termThemeNameAtom);
const defaultFontSize = globalStore.get(getSettingsKeyAtom("term:fontsize")) ?? 12;
const blockData = globalStore.get(this.blockAtom);
const overrideFontSize = blockData?.meta?.["term:fontsize"];
termThemeKeys.sort((a, b) => {
2024-11-09 00:46:44 +01:00
return (termThemes[a]["display:order"] ?? 0) - (termThemes[b]["display:order"] ?? 0);
});
const fullMenu: ContextMenuItem[] = [];
const submenu: ContextMenuItem[] = termThemeKeys.map((themeName) => {
return {
label: termThemes[themeName]["display:name"] ?? themeName,
2024-10-24 20:01:39 +02:00
type: "checkbox",
checked: curThemeName == themeName,
click: () => this.setTerminalTheme(themeName),
};
});
2024-10-24 20:01:39 +02:00
const fontSizeSubMenu: ContextMenuItem[] = [6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18].map(
(fontSize: number) => {
return {
label: fontSize.toString() + "px",
type: "checkbox",
checked: overrideFontSize == fontSize,
click: () => {
2024-10-25 02:02:35 +02:00
RpcApi.SetMetaCommand(TabRpcClient, {
2024-10-24 20:01:39 +02:00
oref: WOS.makeORef("block", this.blockId),
meta: { "term:fontsize": fontSize },
});
},
};
}
);
fontSizeSubMenu.unshift({
label: "Default (" + defaultFontSize + "px)",
type: "checkbox",
checked: overrideFontSize == null,
click: () => {
2024-10-25 02:02:35 +02:00
RpcApi.SetMetaCommand(TabRpcClient, {
2024-10-24 20:01:39 +02:00
oref: WOS.makeORef("block", this.blockId),
meta: { "term:fontsize": null },
});
},
});
fullMenu.push({
label: "Themes",
submenu: submenu,
});
2024-10-24 20:01:39 +02:00
fullMenu.push({
label: "Font Size",
submenu: fontSizeSubMenu,
});
fullMenu.push({ type: "separator" });
fullMenu.push({
label: "Force Restart Controller",
click: this.forceRestartController.bind(this),
});
2024-11-11 22:11:09 +01:00
if (blockData?.meta?.["term:vdomtoolbarblockid"]) {
fullMenu.push({ type: "separator" });
fullMenu.push({
label: "Close Toolbar",
click: () => {
RpcApi.DeleteSubBlockCommand(TabRpcClient, { blockid: blockData.meta["term:vdomtoolbarblockid"] });
},
});
}
return fullMenu;
}
}
2024-10-26 03:36:09 +02:00
function makeTerminalModel(blockId: string, nodeModel: BlockNodeModel): TermViewModel {
2024-10-17 23:50:36 +02:00
return new TermViewModel(blockId, nodeModel);
}
interface TerminalViewProps {
blockId: string;
model: TermViewModel;
}
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-11-11 22:11:09 +01:00
const TermVDomToolbarNode = ({ vdomBlockId, blockId, model }: TerminalViewProps & { vdomBlockId: string }) => {
React.useEffect(() => {
const unsub = waveEventSubscribe({
eventType: "blockclose",
scope: WOS.makeORef("block", vdomBlockId),
handler: (event) => {
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", blockId),
meta: {
"term:mode": null,
"term:vdomtoolbarblockid": null,
},
});
},
});
return () => {
unsub();
};
}, []);
let vdomNodeModel = {
blockId: vdomBlockId,
isFocused: jotai.atom(false),
focusNode: () => {},
onClose: () => {
if (vdomBlockId != null) {
RpcApi.DeleteSubBlockCommand(TabRpcClient, { blockid: vdomBlockId });
}
},
};
const toolbarTarget = jotai.useAtomValue(model.vdomToolbarTarget);
const heightStr = toolbarTarget?.height ?? "1.5em";
return (
<div key="vdomToolbar" className="term-toolbar" style={{ height: heightStr }}>
<SubBlock key="vdom" nodeModel={vdomNodeModel} />
</div>
);
};
2024-10-24 07:47:29 +02:00
const TermVDomNodeSingleId = ({ vdomBlockId, blockId, model }: TerminalViewProps & { vdomBlockId: string }) => {
React.useEffect(() => {
const unsub = waveEventSubscribe({
eventType: "blockclose",
scope: WOS.makeORef("block", vdomBlockId),
handler: (event) => {
RpcApi.SetMetaCommand(TabRpcClient, {
oref: WOS.makeORef("block", blockId),
meta: {
"term:mode": null,
"term:vdomblockid": null,
},
});
},
});
return () => {
unsub();
};
}, []);
const isFocusedAtom = jotai.atom((get) => {
return get(model.nodeModel.isFocused) && get(model.termMode) == "vdom";
});
let vdomNodeModel = {
blockId: vdomBlockId,
isFocused: isFocusedAtom,
2024-10-26 03:36:09 +02:00
focusNode: () => {
model.nodeModel.focusNode();
},
2024-10-24 07:47:29 +02:00
onClose: () => {
if (vdomBlockId != null) {
RpcApi.DeleteSubBlockCommand(TabRpcClient, { blockid: vdomBlockId });
}
},
};
return (
<div key="htmlElem" className="term-htmlelem">
2024-10-26 03:36:09 +02:00
<SubBlock key="vdom" nodeModel={vdomNodeModel} />
2024-10-24 07:47:29 +02:00
</div>
);
};
const TermVDomNode = ({ blockId, model }: TerminalViewProps) => {
const vdomBlockId = jotai.useAtomValue(model.vdomBlockId);
if (vdomBlockId == null) {
return null;
}
return <TermVDomNodeSingleId key={vdomBlockId} vdomBlockId={vdomBlockId} blockId={blockId} model={model} />;
};
2024-11-11 22:11:09 +01:00
const TermToolbarVDomNode = ({ blockId, model }: TerminalViewProps) => {
const vdomToolbarBlockId = jotai.useAtomValue(model.vdomToolbarBlockId);
if (vdomToolbarBlockId == null) {
return null;
}
return (
<TermVDomToolbarNode
key={vdomToolbarBlockId}
vdomBlockId={vdomToolbarBlockId}
blockId={blockId}
model={model}
/>
);
};
const TerminalView = ({ blockId, model }: TerminalViewProps) => {
2024-10-17 23:50:36 +02:00
const viewRef = React.useRef<HTMLDivElement>(null);
const connectElemRef = React.useRef<HTMLDivElement>(null);
const termRef = React.useRef<TermWrap>(null);
model.termRef = termRef;
2024-10-17 23:50:36 +02:00
const spstatusRef = React.useRef<string>(null);
model.shellProcStatusRef = spstatusRef;
const [blockData] = WOS.useWaveObjectValue<Block>(WOS.makeORef("block", blockId));
2024-08-28 03:49:49 +02:00
const termSettingsAtom = useSettingsPrefixAtom("term");
const termSettings = jotai.useAtomValue(termSettingsAtom);
2024-10-17 23:50:36 +02:00
let termMode = blockData?.meta?.["term:mode"] ?? "term";
2024-10-24 07:47:29 +02:00
if (termMode != "term" && termMode != "vdom") {
2024-10-17 23:50:36 +02:00
termMode = "term";
}
const termModeRef = React.useRef(termMode);
2024-10-24 20:01:39 +02:00
const termFontSize = jotai.useAtomValue(model.fontSizeAtom);
React.useEffect(() => {
2024-08-28 03:49:49 +02:00
const fullConfig = globalStore.get(atoms.fullConfigAtom);
const termTheme = computeTheme(fullConfig, blockData?.meta?.["term:theme"]);
const themeCopy = { ...termTheme };
themeCopy.background = "#00000000";
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-10-24 20:01:39 +02:00
const wasFocused = termRef.current != null && globalStore.get(model.nodeModel.isFocused);
2024-06-24 23:34:31 +02:00
const termWrap = new TermWrap(
blockId,
connectElemRef.current,
{
theme: themeCopy,
2024-10-24 20:01:39 +02:00
fontSize: termFontSize,
2024-08-28 03:49:49 +02:00
fontFamily: termSettings?.["term:fontfamily"] ?? "Hack",
2024-06-24 23:34:31 +02:00
drawBoldTextInBrightColors: false,
fontWeight: "normal",
fontWeightBold: "bold",
allowTransparency: true,
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
}
);
(window as any).term = termWrap;
termRef.current = termWrap;
2024-06-12 02:42:10 +02:00
const rszObs = new ResizeObserver(() => {
termWrap.handleResize_debounced();
2024-06-12 02:42:10 +02:00
});
rszObs.observe(connectElemRef.current);
termWrap.initTerminal();
2024-10-24 20:01:39 +02:00
if (wasFocused) {
setTimeout(() => {
model.giveFocus();
}, 10);
}
2024-06-07 00:08:39 +02:00
return () => {
termWrap.dispose();
2024-06-29 02:53:35 +02:00
rszObs.disconnect();
2024-06-07 00:08:39 +02:00
};
2024-10-24 20:01:39 +02:00
}, [blockId, termSettings, termFontSize]);
2024-10-17 23:50:36 +02:00
React.useEffect(() => {
2024-10-24 07:47:29 +02:00
if (termModeRef.current == "vdom" && termMode == "term") {
2024-10-17 23:50:36 +02:00
// 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
// 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);
});
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-06-24 23:34:31 +02:00
let stickerConfig = {
charWidth: 8,
charHeight: 16,
rows: termRef.current?.terminal.rows ?? 24,
cols: termRef.current?.terminal.cols ?? 80,
blockId: blockId,
};
return (
<div className={clsx("view-term", "term-mode-" + termMode)} ref={viewRef}>
<TermResyncHandler blockId={blockId} model={model} />
<TermThemeUpdater blockId={blockId} termRef={termRef} />
<TermStickers config={stickerConfig} />
2024-11-11 22:11:09 +01:00
<TermToolbarVDomNode key="vdom-toolbar" blockId={blockId} model={model} />
2024-10-24 07:47:29 +02:00
<TermVDomNode key="vdom" blockId={blockId} model={model} />
2024-11-11 22:11:09 +01:00
<div key="conntectElem" className="term-connectelem" ref={connectElemRef}></div>
</div>
);
};
export { TermViewModel, TerminalView, makeTerminalModel };