mirror of
https://github.com/khairul169/vaulterm.git
synced 2025-04-29 00:59:40 +07:00
204 lines
4.7 KiB
TypeScript
204 lines
4.7 KiB
TypeScript
"use dom";
|
|
|
|
import React, { CSSProperties, forwardRef, useEffect, useRef } from "react";
|
|
import {
|
|
DOMImperativeFactory,
|
|
DOMProps,
|
|
useDOMImperativeHandle,
|
|
IS_DOM,
|
|
} from "expo/dom";
|
|
import { Terminal as XTerm } from "@xterm/xterm";
|
|
import "@xterm/xterm/css/xterm.css";
|
|
import { FitAddon } from "@xterm/addon-fit/src/FitAddon";
|
|
import { AttachAddon } from "@xterm/addon-attach/src/AttachAddon";
|
|
import { JSONValue } from "expo/build/dom/dom.types";
|
|
|
|
type XTermJsProps = {
|
|
onLoad?: () => void;
|
|
dom?: DOMProps;
|
|
style?: CSSProperties;
|
|
wsUrl: string;
|
|
colorScheme?: "light" | "dark";
|
|
};
|
|
|
|
// vscode-snazzy https://github.com/Tyriar/vscode-snazzy
|
|
const darkTheme = {
|
|
foreground: "#eff0eb",
|
|
background: "#282a36",
|
|
selection: "#97979b33",
|
|
black: "#282a36",
|
|
brightBlack: "#686868",
|
|
red: "#ff5c57",
|
|
brightRed: "#ff5c57",
|
|
green: "#5af78e",
|
|
brightGreen: "#5af78e",
|
|
yellow: "#f3f99d",
|
|
brightYellow: "#f3f99d",
|
|
blue: "#57c7ff",
|
|
brightBlue: "#57c7ff",
|
|
magenta: "#ff6ac1",
|
|
brightMagenta: "#ff6ac1",
|
|
cyan: "#9aedfe",
|
|
brightCyan: "#9aedfe",
|
|
white: "#f1f1f0",
|
|
brightWhite: "#eff0eb",
|
|
};
|
|
|
|
const lightTheme = {
|
|
foreground: "#282a36",
|
|
background: "#e5f2ff",
|
|
selection: "#97979b33",
|
|
black: "#eff0eb",
|
|
brightBlack: "#686868",
|
|
red: "#d32f2f",
|
|
brightRed: "#ff5c57",
|
|
green: "#388e3c",
|
|
brightGreen: "#5af78e",
|
|
yellow: "#fbc02d",
|
|
brightYellow: "#f3f99d",
|
|
blue: "#1976d2",
|
|
brightBlue: "#57c7ff",
|
|
magenta: "#ab47bc",
|
|
brightMagenta: "#ff6ac1",
|
|
cyan: "#00acc1",
|
|
brightCyan: "#9aedfe",
|
|
white: "#282a36",
|
|
brightWhite: "#686868",
|
|
};
|
|
|
|
export interface XTermRef extends DOMImperativeFactory {
|
|
send: (...args: JSONValue[]) => void;
|
|
}
|
|
|
|
const XTermJs = forwardRef<XTermRef, XTermJsProps>((props, ref) => {
|
|
const containerRef = useRef<HTMLDivElement | null>(null);
|
|
const wsRef = useRef<WebSocket | null>(null);
|
|
const xtermRef = useRef<XTerm | null>(null);
|
|
const { wsUrl, onLoad, style = {}, colorScheme } = props;
|
|
const theme = colorScheme === "dark" ? darkTheme : lightTheme;
|
|
|
|
useEffect(() => {
|
|
const container = containerRef.current;
|
|
if (!container) {
|
|
return;
|
|
}
|
|
|
|
const xterm = new XTerm({
|
|
fontFamily: '"Cascadia Code", Menlo, monospace',
|
|
theme: theme,
|
|
cursorBlink: true,
|
|
});
|
|
xterm.open(container);
|
|
xtermRef.current = xterm;
|
|
|
|
const fitAddon = new FitAddon();
|
|
xterm.loadAddon(fitAddon);
|
|
fitAddon.fit();
|
|
|
|
const ws = new WebSocket(wsUrl);
|
|
wsRef.current = ws;
|
|
|
|
const attachAddon = new AttachAddon(ws);
|
|
xterm.loadAddon(attachAddon);
|
|
|
|
if (xterm.element) {
|
|
xterm.element.style.height = "100%";
|
|
}
|
|
|
|
if (IS_DOM) {
|
|
xterm.focus();
|
|
}
|
|
|
|
function resizeTerminal() {
|
|
const { cols, rows } = xterm;
|
|
if (ws.readyState === ws.OPEN) {
|
|
ws.send(`\x01${cols},${rows}`);
|
|
}
|
|
}
|
|
|
|
function onResize() {
|
|
fitAddon.fit();
|
|
resizeTerminal();
|
|
}
|
|
|
|
function onOpen() {
|
|
resizeTerminal();
|
|
}
|
|
|
|
function onClose(e: CloseEvent) {
|
|
// Check if the close event was abnormal
|
|
if (!e.wasClean) {
|
|
const reason = e.reason || `Code: ${e.code}`;
|
|
xterm.write(`\r\nConnection closed unexpectedly: ${reason}\r\n`);
|
|
} else {
|
|
xterm.write("\r\nConnection closed.\r\n");
|
|
}
|
|
}
|
|
|
|
ws.addEventListener("open", onOpen);
|
|
ws.addEventListener("close", onClose);
|
|
xterm.onResize(resizeTerminal);
|
|
window.addEventListener("resize", onResize);
|
|
|
|
// Hack: trigger scale update on visibility change
|
|
const observer = new ResizeObserver(([entry]) => {
|
|
if (entry.contentRect.width > 0) {
|
|
fitAddon.fit();
|
|
}
|
|
});
|
|
observer.observe(containerRef.current!);
|
|
|
|
onLoad?.();
|
|
|
|
return () => {
|
|
xterm.dispose();
|
|
xtermRef.current = null;
|
|
wsRef.current = null;
|
|
containerRef.current = null;
|
|
|
|
ws.close();
|
|
ws.removeEventListener("open", onOpen);
|
|
ws.removeEventListener("close", onClose);
|
|
window.removeEventListener("resize", onResize);
|
|
|
|
observer.disconnect();
|
|
};
|
|
}, [wsUrl]);
|
|
|
|
useDOMImperativeHandle(
|
|
ref,
|
|
() => ({
|
|
send: (...args) => {
|
|
const ws = wsRef.current;
|
|
if (ws?.readyState === ws?.OPEN && args[0]) {
|
|
ws?.send(String(args[0]));
|
|
}
|
|
},
|
|
}),
|
|
[]
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (xtermRef.current) {
|
|
xtermRef.current.options.theme = theme;
|
|
}
|
|
}, [theme]);
|
|
|
|
return (
|
|
<div
|
|
ref={containerRef}
|
|
style={{
|
|
background: theme.background,
|
|
padding: 12,
|
|
flex: !IS_DOM ? 1 : undefined,
|
|
width: "100%",
|
|
height: IS_DOM ? "100vh" : undefined,
|
|
overflow: "hidden",
|
|
...style,
|
|
}}
|
|
/>
|
|
);
|
|
});
|
|
|
|
export default XTermJs;
|