import React, { useEffect, useMemo, useRef, useState } from 'react'; import { ITerminalOptions, Terminal } from 'xterm'; import { FitAddon } from 'xterm-addon-fit'; import { SearchAddon } from 'xterm-addon-search'; import { SearchBarAddon } from 'xterm-addon-search-bar'; import { WebLinksAddon } from 'xterm-addon-web-links'; import { ScrollDownHelperAddon } from '@/plugins/XtermScrollDownHelperAddon'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import { ServerContext } from '@/state/server'; import { usePermissions } from '@/plugins/usePermissions'; import { theme as th } from 'twin.macro'; import useEventListener from '@/plugins/useEventListener'; import { debounce } from 'debounce'; import { usePersistedState } from '@/plugins/usePersistedState'; import { SocketEvent, SocketRequest } from '@/components/server/events'; import classNames from 'classnames'; import { ChevronDoubleRightIcon } from '@heroicons/react/solid'; import CommandRow from '@/blueprint/components/Server/Terminal/CommandRow'; import 'xterm/css/xterm.css'; import styles from './style.module.css'; const theme = { background: th`colors.black`.toString(), cursor: 'transparent', black: th`colors.black`.toString(), 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', selection: '#FAF089', }; const terminalProps: ITerminalOptions = { disableStdin: true, cursorStyle: 'underline', allowTransparency: true, fontSize: 12, fontFamily: th('fontFamily.mono'), rows: 30, theme: theme, }; export default () => { const TERMINAL_PRELUDE = '\u001b[1m\u001b[33mcontainer@pterodactyl~ \u001b[0m'; const ref = useRef(null); const terminal = useMemo(() => new Terminal({ ...terminalProps }), []); const fitAddon = new FitAddon(); const searchAddon = new SearchAddon(); const searchBar = new SearchBarAddon({ searchAddon }); const webLinksAddon = new WebLinksAddon(); const scrollDownHelperAddon = new ScrollDownHelperAddon(); const { connected, instance } = ServerContext.useStoreState((state) => state.socket); const [canSendCommands] = usePermissions(['control.console']); const serverId = ServerContext.useStoreState((state) => state.server.data!.id); const isTransferring = ServerContext.useStoreState((state) => state.server.data!.isTransferring); const [history, setHistory] = usePersistedState(`${serverId}:command_history`, []); const [historyIndex, setHistoryIndex] = useState(-1); // SearchBarAddon has hardcoded z-index: 999 :( const zIndex = ` .xterm-search-bar__addon { z-index: 10; }`; const handleConsoleOutput = (line: string, prelude = false) => terminal.writeln((prelude ? TERMINAL_PRELUDE : '') + line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m'); const handleTransferStatus = (status: string) => { switch (status) { // Sent by either the source or target node if a failure occurs. case 'failure': terminal.writeln(TERMINAL_PRELUDE + 'Transfer has failed.\u001b[0m'); return; } }; const handleDaemonErrorOutput = (line: string) => terminal.writeln( TERMINAL_PRELUDE + '\u001b[1m\u001b[41m' + line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m' ); const handlePowerChangeEvent = (state: string) => terminal.writeln(TERMINAL_PRELUDE + 'Server marked as ' + state + '...\u001b[0m'); const handleCommandKeyDown = (e: React.KeyboardEvent) => { if (e.key === 'ArrowUp') { const newIndex = Math.min(historyIndex + 1, history!.length - 1); setHistoryIndex(newIndex); e.currentTarget.value = history![newIndex] || ''; // By default up arrow will also bring the cursor to the start of the line, // so we'll preventDefault to keep it at the end. e.preventDefault(); } if (e.key === 'ArrowDown') { const newIndex = Math.max(historyIndex - 1, -1); setHistoryIndex(newIndex); e.currentTarget.value = history![newIndex] || ''; } const command = e.currentTarget.value; if (e.key === 'Enter' && command.length > 0) { setHistory((prevHistory) => [command, ...prevHistory!].slice(0, 32)); setHistoryIndex(-1); instance && instance.send('send command', command); e.currentTarget.value = ''; } }; useEffect(() => { if (connected && ref.current && !terminal.element) { terminal.loadAddon(fitAddon); terminal.loadAddon(searchAddon); terminal.loadAddon(searchBar); terminal.loadAddon(webLinksAddon); terminal.loadAddon(scrollDownHelperAddon); terminal.open(ref.current); fitAddon.fit(); searchBar.addNewStyle(zIndex); // Add support for capturing keys terminal.attachCustomKeyEventHandler((e: KeyboardEvent) => { if ((e.ctrlKey || e.metaKey) && e.key === 'c') { document.execCommand('copy'); return false; } else if ((e.ctrlKey || e.metaKey) && e.key === 'f') { e.preventDefault(); searchBar.show(); return false; } else if (e.key === 'Escape') { searchBar.hidden(); } return true; }); } }, [terminal, connected]); useEventListener( 'resize', debounce(() => { if (terminal.element) { fitAddon.fit(); } }, 100) ); useEffect(() => { const listeners: Record void> = { [SocketEvent.STATUS]: handlePowerChangeEvent, [SocketEvent.CONSOLE_OUTPUT]: handleConsoleOutput, [SocketEvent.INSTALL_OUTPUT]: handleConsoleOutput, [SocketEvent.TRANSFER_LOGS]: handleConsoleOutput, [SocketEvent.TRANSFER_STATUS]: handleTransferStatus, [SocketEvent.DAEMON_MESSAGE]: (line) => handleConsoleOutput(line, true), [SocketEvent.DAEMON_ERROR]: handleDaemonErrorOutput, }; if (connected && instance) { // Do not clear the console if the server is being transferred. if (!isTransferring) { terminal.clear(); } Object.keys(listeners).forEach((key: string) => { instance.addListener(key, listeners[key]); }); instance.send(SocketRequest.SEND_LOGS); } return () => { if (instance) { Object.keys(listeners).forEach((key: string) => { instance.removeListener(key, listeners[key]); }); } }; }, [connected, instance]); return (
{canSendCommands && (
)}
); };