mirror of
https://github.com/yudai/gotty.git
synced 2025-12-29 05:38:49 +01:00
116 lines
2.9 KiB
TypeScript
116 lines
2.9 KiB
TypeScript
import {lib} from "libapps";
|
|
|
|
import {IDisposable, Terminal} from 'xterm';
|
|
import {FitAddon} from "xterm-addon-fit";
|
|
|
|
|
|
export class Xterm {
|
|
elem: HTMLElement;
|
|
term: Terminal;
|
|
resizeListener: () => void;
|
|
decoder: lib.UTF8Decoder;
|
|
|
|
message: HTMLElement;
|
|
messageTimeout: number;
|
|
messageTimer: NodeJS.Timer;
|
|
|
|
fitAddon: FitAddon;
|
|
disposables: IDisposable[] = [];
|
|
|
|
|
|
constructor(elem: HTMLElement) {
|
|
this.elem = elem;
|
|
const isWindows = ['Windows', 'Win16', 'Win32', 'WinCE'].indexOf(navigator.platform) >= 0;
|
|
this.term = new Terminal({
|
|
cursorStyle: "block",
|
|
cursorBlink: true,
|
|
windowsMode: isWindows,
|
|
fontFamily: "DejaVu Sans Mono, Everson Mono, FreeMono, Menlo, Terminal, monospace, Apple Symbols",
|
|
fontSize: 12,
|
|
});
|
|
|
|
this.fitAddon = new FitAddon();
|
|
this.term.loadAddon(this.fitAddon);
|
|
|
|
this.message = elem.ownerDocument.createElement("div");
|
|
this.message.className = "xterm-overlay";
|
|
this.messageTimeout = 2000;
|
|
|
|
this.resizeListener = () => {
|
|
this.fitAddon.fit();
|
|
this.term.scrollToBottom();
|
|
this.showMessage(String(this.term.cols) + "x" + String(this.term.rows), this.messageTimeout);
|
|
};
|
|
|
|
this.term.open(elem);
|
|
|
|
this.term.focus()
|
|
this.resizeListener();
|
|
window.addEventListener("resize", () => { this.resizeListener(); });
|
|
|
|
this.decoder = new lib.UTF8Decoder()
|
|
};
|
|
|
|
info(): { columns: number, rows: number } {
|
|
return { columns: this.term.cols, rows: this.term.rows };
|
|
};
|
|
|
|
output(data: string) {
|
|
this.term.write(this.decoder.decode(data));
|
|
};
|
|
|
|
showMessage(message: string, timeout: number) {
|
|
this.message.textContent = message;
|
|
this.elem.appendChild(this.message);
|
|
|
|
if (this.messageTimer) {
|
|
clearTimeout(this.messageTimer);
|
|
}
|
|
if (timeout > 0) {
|
|
this.messageTimer = setTimeout(() => {
|
|
this.elem.removeChild(this.message);
|
|
}, timeout);
|
|
}
|
|
};
|
|
|
|
removeMessage(): void {
|
|
if (this.message.parentNode == this.elem) {
|
|
this.elem.removeChild(this.message);
|
|
}
|
|
}
|
|
|
|
setWindowTitle(title: string) {
|
|
document.title = title;
|
|
};
|
|
|
|
setPreferences(value: object) {
|
|
};
|
|
|
|
onInput(callback: (input: string) => void) {
|
|
this.disposables.push(this.term.onData((data) => {
|
|
callback(data);
|
|
}));
|
|
|
|
};
|
|
|
|
onResize(callback: (colmuns: number, rows: number) => void) {
|
|
this.disposables.push(this.term.onResize((data) => {
|
|
callback(data.cols, data.rows);
|
|
}));
|
|
};
|
|
|
|
deactivate(): void {
|
|
this.disposables.forEach(d => d.dispose())
|
|
this.term.blur();
|
|
}
|
|
|
|
reset(): void {
|
|
this.removeMessage();
|
|
this.term.clear();
|
|
}
|
|
|
|
close(): void {
|
|
window.removeEventListener("resize", this.resizeListener);
|
|
this.term.dispose();
|
|
}
|
|
}
|