Compare commits
4 Commits
521fb8f613
...
648fb2778c
| Author | SHA1 | Date | |
|---|---|---|---|
| 648fb2778c | |||
| ae6acc3bee | |||
| 3705e72b6d | |||
| 0fce95e9b8 |
|
|
@ -11,6 +11,8 @@ export default function SerialConsolePage({ onBack }: Props) {
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
const termRef = useRef<Terminal | null>(null);
|
const termRef = useRef<Terminal | null>(null);
|
||||||
const fitRef = useRef<FitAddon | null>(null);
|
const fitRef = useRef<FitAddon | null>(null);
|
||||||
|
const lineBuffer = useRef('');
|
||||||
|
const echoQueue = useRef<string[]>([]);
|
||||||
const { status, send, subscribe } = useWs();
|
const { status, send, subscribe } = useWs();
|
||||||
|
|
||||||
// Keep a stable ref to `send` so the xterm onData closure never goes stale
|
// Keep a stable ref to `send` so the xterm onData closure never goes stale
|
||||||
|
|
@ -46,7 +48,28 @@ export default function SerialConsolePage({ onBack }: Props) {
|
||||||
term.writeln('\x1b[2m── Meatloaf Serial Console ──\x1b[0m');
|
term.writeln('\x1b[2m── Meatloaf Serial Console ──\x1b[0m');
|
||||||
term.writeln('');
|
term.writeln('');
|
||||||
|
|
||||||
term.onData((data) => sendRef.current(data));
|
term.onData((data) => {
|
||||||
|
for (const char of data) {
|
||||||
|
if (char === '\r' || char === '\n') {
|
||||||
|
const line = lineBuffer.current + '\r';
|
||||||
|
lineBuffer.current = '';
|
||||||
|
term.write('\r\n');
|
||||||
|
echoQueue.current.push(line);
|
||||||
|
sendRef.current(line);
|
||||||
|
} else if (char === '\x7f' || char === '\b') {
|
||||||
|
if (lineBuffer.current.length > 0) {
|
||||||
|
lineBuffer.current = lineBuffer.current.slice(0, -1);
|
||||||
|
term.write('\b \b');
|
||||||
|
}
|
||||||
|
} else if (char === '\x03') {
|
||||||
|
lineBuffer.current = '';
|
||||||
|
term.write('^C\r\n');
|
||||||
|
} else if (char >= ' ') {
|
||||||
|
lineBuffer.current += char;
|
||||||
|
term.write(char);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const onResize = () => fitRef.current?.fit();
|
const onResize = () => fitRef.current?.fit();
|
||||||
window.addEventListener('resize', onResize);
|
window.addEventListener('resize', onResize);
|
||||||
|
|
@ -59,9 +82,14 @@ export default function SerialConsolePage({ onBack }: Props) {
|
||||||
};
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
// Forward all incoming WS messages to the terminal
|
// Forward all incoming WS messages to the terminal, suppressing our own echoes
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
return subscribe((msg) => {
|
return subscribe((msg) => {
|
||||||
|
const idx = echoQueue.current.indexOf(msg);
|
||||||
|
if (idx !== -1) {
|
||||||
|
echoQueue.current.splice(idx, 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
termRef.current?.write(msg);
|
termRef.current?.write(msg);
|
||||||
});
|
});
|
||||||
}, [subscribe]);
|
}, [subscribe]);
|
||||||
|
|
@ -109,10 +137,16 @@ export default function SerialConsolePage({ onBack }: Props) {
|
||||||
|
|
||||||
{/* Terminal */}
|
{/* Terminal */}
|
||||||
<div
|
<div
|
||||||
ref={containerRef}
|
className="flex-1 min-h-0 relative overflow-hidden"
|
||||||
className="flex-1 min-h-0"
|
|
||||||
style={{ paddingBottom: 'env(safe-area-inset-bottom)' }}
|
style={{ paddingBottom: 'env(safe-area-inset-bottom)' }}
|
||||||
/>
|
>
|
||||||
|
<div ref={containerRef} className="absolute inset-0" />
|
||||||
|
<div
|
||||||
|
className="absolute inset-0 pointer-events-none z-10 opacity-15"
|
||||||
|
style={{ backgroundImage: 'url(assets/icon.svg)', backgroundRepeat: 'repeat', backgroundSize: '64px 64px' }}
|
||||||
|
aria-hidden="true"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue
Block a user