Still completely broken terminal...

This commit is contained in:
Dane Everitt 2019-09-05 23:05:24 -07:00
parent 54339c1344
commit 1ae374069c
No known key found for this signature in database
GPG key ID: EEA66103B3D71F53

View file

@ -1,10 +1,8 @@
import React, { createRef } from 'react'; import React, { createRef, useEffect } from 'react';
import { Terminal } from 'xterm'; import { Terminal } from 'xterm';
import * as TerminalFit from 'xterm/lib/addons/fit/fit'; import * as TerminalFit from 'xterm/lib/addons/fit/fit';
import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
import { connect } from 'react-redux'; import { ServerContext } from '@/state/server';
import { Websocket } from '@/plugins/Websocket';
import { ServerStore } from '@/state/server';
const theme = { const theme = {
background: 'transparent', background: 'transparent',
@ -27,94 +25,71 @@ const theme = {
brightWhite: '#ffffff', brightWhite: '#ffffff',
}; };
interface Props { const terminal = new Terminal({
connected: boolean; disableStdin: true,
instance: Websocket | null; cursorStyle: 'underline',
} allowTransparency: true,
fontSize: 12,
fontFamily: 'Menlo, Monaco, Consolas, monospace',
rows: 30,
theme: theme,
});
class Console extends React.PureComponent<Readonly<Props>> { export default () => {
ref = createRef<HTMLDivElement>(); const ref = createRef<HTMLDivElement>();
terminal = new Terminal({ const connected = ServerContext.useStoreState(state => state.socket.connected);
disableStdin: true, const instance = ServerContext.useStoreState(state => state.socket.instance);
cursorStyle: 'underline',
allowTransparency: true,
fontSize: 12,
fontFamily: 'Menlo, Monaco, Consolas, monospace',
rows: 30,
theme: theme,
});
componentDidMount () { const handleConsoleOutput = (line: string) => terminal.writeln(
if (this.ref.current) { line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m',
this.terminal.open(this.ref.current); );
this.terminal.clear();
useEffect(() => {
if (ref.current) {
terminal.open(ref.current);
// @see https://github.com/xtermjs/xterm.js/issues/2265 // @see https://github.com/xtermjs/xterm.js/issues/2265
// @see https://github.com/xtermjs/xterm.js/issues/2230 // @see https://github.com/xtermjs/xterm.js/issues/2230
TerminalFit.fit(this.terminal); 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');
} }
if (this.props.connected && this.props.instance) { return () => {
this.listenForEvents(); instance && instance
} .removeListener('console output', handleConsoleOutput)
} .removeAllListeners('stats');
};
}, [ connected, instance ]);
componentDidUpdate (prevProps: Readonly<Readonly<Props>>) { return (
if (!prevProps.connected && this.props.connected) { <div className={'text-xs font-mono relative'}>
this.listenForEvents(); <SpinnerOverlay visible={!connected} size={'large'}/>
} <div
} className={'rounded-t p-2 bg-black overflow-scroll w-full'}
style={{
componentWillUnmount () { minHeight: '16rem',
if (this.props.instance) { maxHeight: '64rem',
this.props.instance.removeListener('server log', this.handleServerLog); }}
this.props.instance.removeListener('server log', this.handleConsoleOutput); >
} <div id={'terminal'} ref={ref}/>
} </div>
<div className={'rounded-b bg-neutral-900 text-neutral-100 flex'}>
listenForEvents () { <div className={'flex-no-shrink p-2 font-bold'}>$</div>
const instance = this.props.instance!; <div className={'w-full'}>
<input type={'text'} className={'bg-transparent text-neutral-100 p-2 pl-0 w-full'}/>
instance.addListener('server log', this.handleServerLog);
instance.addListener('console output', this.handleConsoleOutput);
instance.send('send logs');
}
handleServerLog = (lines: string[]) => lines.forEach(data => {
return data.split(/\n/g).forEach(line => this.terminal.writeln(line + '\u001b[0m'));
});
handleConsoleOutput = (line: string) => this.terminal.writeln(
line.replace(/(?:\r\n|\r|\n)$/im, '') + '\u001b[0m'
);
render () {
return (
<div className={'text-xs font-mono relative'}>
<SpinnerOverlay visible={!this.props.connected} size={'large'}/>
<div
className={'rounded-t p-2 bg-black overflow-scroll w-full'}
style={{
minHeight: '16rem',
maxHeight: '64rem',
}}
>
<div id={'terminal'} ref={this.ref}/>
</div>
<div className={'rounded-b bg-neutral-900 text-neutral-100 flex'}>
<div className={'flex-no-shrink p-2 font-bold'}>$</div>
<div className={'w-full'}>
<input type={'text'} className={'bg-transparent text-neutral-100 p-2 pl-0 w-full'}/>
</div>
</div> </div>
</div> </div>
); </div>
} );
} };
export default connect(
(state: ServerStore) => ({
connected: state.socket.connected,
instance: state.socket.instance,
}),
)(Console);