2024-05-14 08:45:41 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-07-25 05:34:22 +02:00
|
|
|
import { WshServer } from "@/app/store/wshserver";
|
|
|
|
import { VDomView } from "@/app/view/term/vdom";
|
2024-07-18 00:24:43 +02:00
|
|
|
import { WOS, atoms, getEventORefSubject, globalStore, useBlockAtom, useSettingsAtom } 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 { produce } from "immer";
|
|
|
|
import * as jotai from "jotai";
|
2024-08-07 23:27:16 +02:00
|
|
|
import "public/xterm.css";
|
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-05-14 08:45:41 +02:00
|
|
|
|
2024-06-14 08:54:04 +02:00
|
|
|
const keyMap = {
|
|
|
|
Enter: "\r",
|
|
|
|
Backspace: "\x7f",
|
|
|
|
Tab: "\t",
|
|
|
|
Escape: "\x1b",
|
|
|
|
ArrowUp: "\x1b[A",
|
|
|
|
ArrowDown: "\x1b[B",
|
|
|
|
ArrowRight: "\x1b[C",
|
|
|
|
ArrowLeft: "\x1b[D",
|
|
|
|
Insert: "\x1b[2~",
|
|
|
|
Delete: "\x1b[3~",
|
|
|
|
Home: "\x1b[1~",
|
|
|
|
End: "\x1b[4~",
|
|
|
|
PageUp: "\x1b[5~",
|
|
|
|
PageDown: "\x1b[6~",
|
|
|
|
};
|
|
|
|
|
|
|
|
function keyboardEventToASCII(event: React.KeyboardEvent<HTMLInputElement>): string {
|
|
|
|
// check modifiers
|
|
|
|
// if no modifiers are set, just send the key
|
|
|
|
if (!event.altKey && !event.ctrlKey && !event.metaKey) {
|
|
|
|
if (event.key == null || event.key == "") {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
if (keyMap[event.key] != null) {
|
|
|
|
return keyMap[event.key];
|
|
|
|
}
|
|
|
|
if (event.key.length == 1) {
|
|
|
|
return event.key;
|
|
|
|
} else {
|
|
|
|
console.log("not sending keyboard event", event.key, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if meta or alt is set, there is no ASCII representation
|
|
|
|
if (event.metaKey || event.altKey) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
// if ctrl is set, if it is a letter, subtract 64 from the uppercase value to get the ASCII value
|
|
|
|
if (event.ctrlKey) {
|
|
|
|
if (
|
|
|
|
(event.key.length === 1 && event.key >= "A" && event.key <= "Z") ||
|
|
|
|
(event.key >= "a" && event.key <= "z")
|
|
|
|
) {
|
|
|
|
const key = event.key.toUpperCase();
|
|
|
|
return String.fromCharCode(key.charCodeAt(0) - 64);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2024-05-29 09:28:25 +02:00
|
|
|
type InitialLoadDataType = {
|
|
|
|
loaded: boolean;
|
|
|
|
heldData: Uint8Array[];
|
|
|
|
};
|
|
|
|
|
2024-07-23 22:16:53 +02:00
|
|
|
function vdomText(text: string): VDomElem {
|
|
|
|
return {
|
|
|
|
tag: "#text",
|
|
|
|
text: text,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const testVDom: VDomElem = {
|
|
|
|
id: "testid1",
|
2024-06-14 08:54:04 +02:00
|
|
|
tag: "div",
|
|
|
|
children: [
|
|
|
|
{
|
2024-07-25 05:18:55 +02:00
|
|
|
id: "testh1",
|
2024-06-14 08:54:04 +02:00
|
|
|
tag: "h1",
|
2024-07-23 22:16:53 +02:00
|
|
|
children: [vdomText("Hello World")],
|
2024-06-14 08:54:04 +02:00
|
|
|
},
|
|
|
|
{
|
2024-07-25 05:18:55 +02:00
|
|
|
id: "testp",
|
2024-06-14 08:54:04 +02:00
|
|
|
tag: "p",
|
2024-07-23 22:16:53 +02:00
|
|
|
children: [vdomText("This is a paragraph (from VDOM)")],
|
2024-06-14 08:54:04 +02:00
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
|
|
|
|
function setBlockFocus(blockId: string) {
|
|
|
|
let winData = globalStore.get(atoms.waveWindow);
|
2024-07-03 23:31:02 +02:00
|
|
|
if (winData == null) {
|
|
|
|
return;
|
|
|
|
}
|
2024-06-14 08:54:04 +02:00
|
|
|
winData = produce(winData, (draft) => {
|
|
|
|
draft.activeblockid = blockId;
|
|
|
|
});
|
|
|
|
WOS.setObjectValue(winData, globalStore.set, true);
|
|
|
|
}
|
|
|
|
|
2024-07-23 02:08:18 +02:00
|
|
|
class TermViewModel {
|
|
|
|
termRef: React.RefObject<TermWrap>;
|
|
|
|
blockAtom: jotai.Atom<Block>;
|
|
|
|
termMode: jotai.Atom<string>;
|
|
|
|
htmlElemFocusRef: React.RefObject<HTMLInputElement>;
|
|
|
|
blockId: string;
|
2024-07-26 01:45:07 +02:00
|
|
|
viewIcon: jotai.Atom<string>;
|
|
|
|
viewText: 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-07-23 02:08:18 +02:00
|
|
|
|
|
|
|
constructor(blockId: string) {
|
|
|
|
this.blockId = blockId;
|
|
|
|
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-07-26 01:45:07 +02:00
|
|
|
this.viewText = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
return blockData?.meta?.title ?? "";
|
|
|
|
});
|
2024-07-31 04:52:50 +02:00
|
|
|
this.blockBg = jotai.atom((get) => {
|
|
|
|
const blockData = get(this.blockAtom);
|
|
|
|
const settings = globalStore.get(atoms.settingsConfigAtom);
|
|
|
|
const theme = computeTheme(settings, blockData?.meta?.["term:theme"]);
|
|
|
|
if (theme != null && theme.background != null) {
|
|
|
|
return { bg: theme.background };
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
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;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (this.htmlElemFocusRef?.current) {
|
|
|
|
this.htmlElemFocusRef.current.focus();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2024-07-31 04:52:50 +02:00
|
|
|
|
|
|
|
setTerminalTheme(themeName: string) {
|
|
|
|
WshServer.SetMetaCommand({ oref: WOS.makeORef("block", this.blockId), meta: { "term:theme": themeName } });
|
|
|
|
}
|
|
|
|
|
|
|
|
getSettingsMenuItems(): ContextMenuItem[] {
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
label: "Themes",
|
|
|
|
submenu: [
|
|
|
|
{ label: "Default Dark", click: () => this.setTerminalTheme("default") },
|
|
|
|
{ label: "Dracula", click: () => this.setTerminalTheme("dracula") },
|
|
|
|
{ label: "Campbell", click: () => this.setTerminalTheme("campbell") },
|
|
|
|
],
|
|
|
|
},
|
|
|
|
];
|
|
|
|
}
|
2024-07-23 02:08:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function makeTerminalModel(blockId: string): TermViewModel {
|
|
|
|
return new TermViewModel(blockId);
|
|
|
|
}
|
|
|
|
|
2024-07-25 05:34:22 +02:00
|
|
|
interface TerminalViewProps {
|
|
|
|
blockId: string;
|
|
|
|
model: TermViewModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
const TerminalView = ({ blockId, model }: TerminalViewProps) => {
|
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-06-24 23:34:31 +02:00
|
|
|
const shellProcStatusRef = React.useRef<string>(null);
|
|
|
|
const blockIconOverrideAtom = useBlockAtom<string>(blockId, "blockicon:override", () => {
|
|
|
|
return jotai.atom<string>(null);
|
|
|
|
}) as jotai.PrimitiveAtom<string>;
|
2024-06-13 23:41:28 +02:00
|
|
|
const htmlElemFocusRef = React.useRef<HTMLInputElement>(null);
|
2024-07-23 02:08:18 +02:00
|
|
|
model.htmlElemFocusRef = htmlElemFocusRef;
|
2024-06-13 23:41:28 +02:00
|
|
|
const [blockData] = WOS.useWaveObjectValue<Block>(WOS.makeORef("block", blockId));
|
2024-06-14 08:54:04 +02:00
|
|
|
const isFocusedAtom = useBlockAtom<boolean>(blockId, "isFocused", () => {
|
|
|
|
return jotai.atom((get) => {
|
|
|
|
const winData = get(atoms.waveWindow);
|
2024-07-03 23:31:02 +02:00
|
|
|
return winData?.activeblockid === blockId;
|
2024-06-14 08:54:04 +02:00
|
|
|
});
|
|
|
|
});
|
2024-06-21 22:23:07 +02:00
|
|
|
const termSettingsAtom = useSettingsAtom<TerminalConfigType>("term", (settings: SettingsConfigType) => {
|
|
|
|
return settings?.term;
|
|
|
|
});
|
|
|
|
const termSettings = jotai.useAtomValue(termSettingsAtom);
|
2024-06-14 08:54:04 +02:00
|
|
|
const isFocused = jotai.useAtomValue(isFocusedAtom);
|
2024-07-25 05:34:22 +02:00
|
|
|
|
2024-05-14 08:45:41 +02:00
|
|
|
React.useEffect(() => {
|
2024-06-24 23:34:31 +02:00
|
|
|
function handleTerminalKeydown(event: KeyboardEvent) {
|
|
|
|
const waveEvent = keyutil.adaptFromReactOrNativeKeyEvent(event);
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Cmd:Escape")) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
2024-07-18 00:24:43 +02:00
|
|
|
WshServer.SetMetaCommand({ oref: WOS.makeORef("block", blockId), meta: { "term:mode": null } });
|
2024-06-24 23:34:31 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (shellProcStatusRef.current != "running" && keyutil.checkKeyPressed(waveEvent, "Enter")) {
|
|
|
|
// restart
|
2024-07-26 22:30:11 +02:00
|
|
|
WshServer.ControllerRestartCommand({ blockid: blockId });
|
2024-06-24 23:34:31 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2024-07-25 05:34:22 +02:00
|
|
|
const settings = globalStore.get(atoms.settingsConfigAtom);
|
2024-07-30 21:33:28 +02:00
|
|
|
const termTheme = computeTheme(settings, blockData?.meta?.["term:theme"]);
|
2024-07-31 04:52:50 +02:00
|
|
|
const themeCopy = { ...termTheme };
|
|
|
|
themeCopy.background = "#00000000";
|
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-06-24 23:34:31 +02:00
|
|
|
fontSize: termSettings?.fontsize ?? 12,
|
|
|
|
fontFamily: termSettings?.fontfamily ?? "Hack",
|
|
|
|
drawBoldTextInBrightColors: false,
|
|
|
|
fontWeight: "normal",
|
|
|
|
fontWeightBold: "bold",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
keydownHandler: handleTerminalKeydown,
|
2024-08-07 23:27:16 +02:00
|
|
|
useWebGl: !termSettings?.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;
|
|
|
|
termWrap.addFocusListener(() => {
|
2024-06-14 08:54:04 +02:00
|
|
|
setBlockFocus(blockId);
|
|
|
|
});
|
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-06-18 07:38:48 +02:00
|
|
|
const handleHtmlKeyDown = (event: React.KeyboardEvent<HTMLInputElement>) => {
|
2024-06-24 23:34:31 +02:00
|
|
|
const waveEvent = keyutil.adaptFromReactOrNativeKeyEvent(event);
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Cmd:Escape")) {
|
2024-06-13 23:41:28 +02:00
|
|
|
// reset term:mode
|
2024-07-18 00:24:43 +02:00
|
|
|
WshServer.SetMetaCommand({ oref: WOS.makeORef("block", blockId), meta: { "term:mode": null } });
|
2024-06-13 23:41:28 +02:00
|
|
|
return false;
|
|
|
|
}
|
2024-06-14 08:54:04 +02:00
|
|
|
const asciiVal = keyboardEventToASCII(event);
|
|
|
|
if (asciiVal.length == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const b64data = btoa(asciiVal);
|
2024-07-26 22:30:11 +02:00
|
|
|
WshServer.ControllerInputCommand({ blockid: blockId, inputdata64: b64data });
|
2024-06-13 23:41:28 +02:00
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
let termMode = blockData?.meta?.["term:mode"] ?? "term";
|
|
|
|
if (termMode != "term" && termMode != "html") {
|
|
|
|
termMode = "term";
|
|
|
|
}
|
2024-06-14 08:54:04 +02:00
|
|
|
|
2024-07-03 23:31:02 +02:00
|
|
|
// set initial focus
|
2024-06-14 08:54:04 +02:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (isFocused && termMode == "term") {
|
2024-06-18 07:38:48 +02:00
|
|
|
termRef.current?.terminal.focus();
|
2024-06-14 08:54:04 +02:00
|
|
|
}
|
|
|
|
if (isFocused && termMode == "html") {
|
|
|
|
htmlElemFocusRef.current?.focus();
|
|
|
|
}
|
2024-07-03 23:31:02 +02:00
|
|
|
}, []);
|
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;
|
|
|
|
}
|
|
|
|
shellProcStatusRef.current = status;
|
|
|
|
if (status == "running") {
|
|
|
|
termRef.current?.setIsRunning(true);
|
2024-07-06 00:47:35 +02:00
|
|
|
globalStore.set(blockIconOverrideAtom, "terminal");
|
2024-06-24 23:34:31 +02:00
|
|
|
} else {
|
|
|
|
termRef.current?.setIsRunning(false);
|
2024-07-06 00:47:35 +02:00
|
|
|
globalStore.set(blockIconOverrideAtom, "regular@terminal");
|
2024-06-24 23:34:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
const initialRTStatus = services.BlockService.GetControllerStatus(blockId);
|
|
|
|
initialRTStatus.then((rts) => {
|
|
|
|
updateShellProcStatus(rts?.shellprocstatus);
|
|
|
|
});
|
|
|
|
const bcSubject = getEventORefSubject("blockcontroller:status", WOS.makeORef("block", blockId));
|
2024-07-03 23:31:02 +02:00
|
|
|
const sub = bcSubject.subscribe((data: WSEventType) => {
|
2024-06-24 23:34:31 +02:00
|
|
|
let bcRTS: BlockControllerRuntimeStatus = data.data;
|
|
|
|
updateShellProcStatus(bcRTS?.shellprocstatus);
|
|
|
|
});
|
2024-07-03 23:31:02 +02:00
|
|
|
return () => sub.unsubscribe();
|
|
|
|
}, []);
|
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-06-25 02:58:40 +02:00
|
|
|
function handleKeyDown(e: React.KeyboardEvent<HTMLDivElement>) {
|
|
|
|
const waveEvent = keyutil.adaptFromReactOrNativeKeyEvent(e);
|
|
|
|
if (keyutil.checkKeyPressed(waveEvent, "Cmd:Shift:v")) {
|
|
|
|
const p = navigator.clipboard.readText();
|
|
|
|
p.then((text) => {
|
|
|
|
termRef.current?.handleTermData(text);
|
|
|
|
});
|
2024-06-25 03:57:01 +02:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2024-06-25 02:58:40 +02:00
|
|
|
return true;
|
|
|
|
} else if (keyutil.checkKeyPressed(waveEvent, "Cmd:Shift:c")) {
|
|
|
|
const sel = termRef.current?.terminal.getSelection();
|
|
|
|
navigator.clipboard.writeText(sel);
|
2024-06-25 03:57:01 +02:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2024-06-25 02:58:40 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-15 01:53:03 +02:00
|
|
|
return (
|
2024-06-25 02:58:40 +02:00
|
|
|
<div
|
|
|
|
className={clsx("view-term", "term-mode-" + termMode, isFocused ? "is-focused" : null)}
|
|
|
|
onKeyDown={handleKeyDown}
|
|
|
|
>
|
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-06-13 23:41:28 +02:00
|
|
|
<div
|
|
|
|
key="htmlElem"
|
|
|
|
className="term-htmlelem"
|
|
|
|
onClick={() => {
|
|
|
|
if (htmlElemFocusRef.current != null) {
|
|
|
|
htmlElemFocusRef.current.focus();
|
|
|
|
}
|
2024-06-14 08:54:04 +02:00
|
|
|
setBlockFocus(blockId);
|
2024-06-13 23:41:28 +02:00
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div key="htmlElemFocus" className="term-htmlelem-focus">
|
2024-06-14 08:54:04 +02:00
|
|
|
<input
|
|
|
|
type="text"
|
|
|
|
value={""}
|
|
|
|
ref={htmlElemFocusRef}
|
2024-06-18 07:38:48 +02:00
|
|
|
onKeyDown={handleHtmlKeyDown}
|
2024-06-14 08:54:04 +02:00
|
|
|
onChange={() => {}}
|
|
|
|
/>
|
2024-06-13 23:41:28 +02:00
|
|
|
</div>
|
|
|
|
<div key="htmlElemContent" className="term-htmlelem-content">
|
2024-07-23 22:16:53 +02:00
|
|
|
<VDomView rootNode={testVDom} />
|
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-07-23 02:08:18 +02:00
|
|
|
export { TerminalView, makeTerminalModel };
|