import React, { createRef, useEffect, useMemo } from 'react'; import { ITerminalOptions, Terminal } from 'xterm'; import * as TerminalFit from 'xterm/lib/addons/fit/fit'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import { ServerContext } from '@/state/server'; const theme = { background: 'transparent', cursor: 'transparent', black: '#000000', red: '#E54B4B', green: '#9ECE58', yellow: '#FAED70', blue: '#396FE2', magenta: '#BB80B3', cyan: '#2DDAFD', white: '#d0d0d0', brightBlack: 'rgba(255, 255, 255, 0.2)', brightRed: '#FF5370', brightGreen: '#C3E88D', brightYellow: '#FFCB6B', brightBlue: '#82AAFF', brightMagenta: '#C792EA', brightCyan: '#89DDFF', brightWhite: '#ffffff', }; const terminalProps: ITerminalOptions = { disableStdin: true, cursorStyle: 'underline', allowTransparency: true, fontSize: 12, fontFamily: 'Menlo, Monaco, Consolas, monospace', rows: 30, theme: theme, }; export default () => { const ref = createRef(); const terminal = useMemo(() => new Terminal({ ...terminalProps }), []); const connected = ServerContext.useStoreState(state => state.socket.connected); const instance = ServerContext.useStoreState(state => state.socket.instance); const handleConsoleOutput = (line: string) => terminal.writeln( line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m', ); useEffect(() => { if (ref.current && !terminal.element) { terminal.open(ref.current); // @see https://github.com/xtermjs/xterm.js/issues/2265 // @see https://github.com/xtermjs/xterm.js/issues/2230 TerminalFit.fit(terminal); } }, [ ref.current ]); useEffect(() => { if (connected && instance) { terminal.clear(); instance // .addListener('stats', data => console.log(JSON.parse(data))) .addListener('console output', handleConsoleOutput); instance.send('send logs'); } return () => { instance && instance .removeAllListeners('console output') .removeAllListeners('stats'); }; }, [ connected, instance ]); return (
$
); };