2024-06-18 07:38:48 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-09-12 03:03:55 +02:00
|
|
|
import { getFileSubject } from "@/app/store/wps";
|
2024-10-10 19:12:42 +02:00
|
|
|
import { sendWSCommand } from "@/app/store/ws";
|
2024-09-16 20:59:39 +02:00
|
|
|
import { RpcApi } from "@/app/store/wshclientapi";
|
2024-10-17 23:34:02 +02:00
|
|
|
import { TabRpcClient } from "@/app/store/wshrpcutil";
|
2024-10-14 19:05:38 +02:00
|
|
|
import { PLATFORM, WOS, atoms, fetchWaveFile, getSettingsKeyAtom, globalStore, openLink } from "@/store/global";
|
2024-06-18 07:38:48 +02:00
|
|
|
import * as services from "@/store/services";
|
2024-08-19 20:54:54 +02:00
|
|
|
import * as util from "@/util/util";
|
2024-08-07 23:27:16 +02:00
|
|
|
import { base64ToArray, fireAndForget } from "@/util/util";
|
2024-06-18 07:38:48 +02:00
|
|
|
import { SerializeAddon } from "@xterm/addon-serialize";
|
2024-08-07 23:27:16 +02:00
|
|
|
import { WebLinksAddon } from "@xterm/addon-web-links";
|
|
|
|
import { WebglAddon } from "@xterm/addon-webgl";
|
2024-06-18 07:38:48 +02:00
|
|
|
import * as TermTypes from "@xterm/xterm";
|
|
|
|
import { Terminal } from "@xterm/xterm";
|
2024-09-05 09:21:08 +02:00
|
|
|
import debug from "debug";
|
2024-06-18 07:38:48 +02:00
|
|
|
import { debounce } from "throttle-debounce";
|
2024-07-09 08:13:12 +02:00
|
|
|
import { FitAddon } from "./fitaddon";
|
2024-06-18 07:38:48 +02:00
|
|
|
|
2024-09-05 09:21:08 +02:00
|
|
|
const dlog = debug("wave:termwrap");
|
|
|
|
|
2024-08-13 00:53:34 +02:00
|
|
|
const TermFileName = "term";
|
|
|
|
const TermCacheFileName = "cache:term:full";
|
2024-10-07 18:51:23 +02:00
|
|
|
const MinDataProcessedForCache = 100 * 1024;
|
2024-08-13 00:53:34 +02:00
|
|
|
|
2024-08-07 23:27:16 +02:00
|
|
|
// detect webgl support
|
|
|
|
function detectWebGLSupport(): boolean {
|
|
|
|
try {
|
|
|
|
const canvas = document.createElement("canvas");
|
|
|
|
const ctx = canvas.getContext("webgl");
|
|
|
|
return !!ctx;
|
|
|
|
} catch (e) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const WebGLSupported = detectWebGLSupport();
|
|
|
|
let loggedWebGL = false;
|
|
|
|
|
|
|
|
type TermWrapOptions = {
|
2024-08-19 21:24:08 +02:00
|
|
|
keydownHandler?: (e: KeyboardEvent) => boolean;
|
2024-08-07 23:27:16 +02:00
|
|
|
useWebGl?: boolean;
|
|
|
|
};
|
|
|
|
|
2024-06-18 07:38:48 +02:00
|
|
|
export class TermWrap {
|
|
|
|
blockId: string;
|
|
|
|
ptyOffset: number;
|
|
|
|
dataBytesProcessed: number;
|
|
|
|
terminal: Terminal;
|
|
|
|
connectElem: HTMLDivElement;
|
|
|
|
fitAddon: FitAddon;
|
|
|
|
serializeAddon: SerializeAddon;
|
2024-07-09 08:13:12 +02:00
|
|
|
mainFileSubject: SubjectWithRef<WSFileEventData>;
|
2024-06-18 07:38:48 +02:00
|
|
|
loaded: boolean;
|
|
|
|
heldData: Uint8Array[];
|
|
|
|
handleResize_debounced: () => void;
|
2024-06-24 23:34:31 +02:00
|
|
|
isRunning: boolean;
|
2024-09-05 09:21:08 +02:00
|
|
|
hasResized: boolean;
|
2024-06-18 07:38:48 +02:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
blockId: string,
|
|
|
|
connectElem: HTMLDivElement,
|
2024-06-24 23:34:31 +02:00
|
|
|
options: TermTypes.ITerminalOptions & TermTypes.ITerminalInitOnlyOptions,
|
2024-08-07 23:27:16 +02:00
|
|
|
waveOptions: TermWrapOptions
|
2024-06-18 07:38:48 +02:00
|
|
|
) {
|
2024-08-16 22:42:16 +02:00
|
|
|
this.loaded = false;
|
2024-06-18 07:38:48 +02:00
|
|
|
this.blockId = blockId;
|
|
|
|
this.ptyOffset = 0;
|
|
|
|
this.dataBytesProcessed = 0;
|
2024-09-05 09:21:08 +02:00
|
|
|
this.hasResized = false;
|
2024-06-18 07:38:48 +02:00
|
|
|
this.terminal = new Terminal(options);
|
|
|
|
this.fitAddon = new FitAddon();
|
2024-07-09 08:13:12 +02:00
|
|
|
this.fitAddon.noScrollbar = PLATFORM == "darwin";
|
2024-06-18 07:38:48 +02:00
|
|
|
this.serializeAddon = new SerializeAddon();
|
|
|
|
this.terminal.loadAddon(this.fitAddon);
|
|
|
|
this.terminal.loadAddon(this.serializeAddon);
|
2024-08-07 23:27:16 +02:00
|
|
|
this.terminal.loadAddon(
|
|
|
|
new WebLinksAddon((e, uri) => {
|
|
|
|
e.preventDefault();
|
|
|
|
switch (PLATFORM) {
|
|
|
|
case "darwin":
|
|
|
|
if (e.metaKey) {
|
|
|
|
fireAndForget(() => openLink(uri));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (e.ctrlKey) {
|
|
|
|
fireAndForget(() => openLink(uri));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
|
|
|
if (WebGLSupported && waveOptions.useWebGl) {
|
|
|
|
const webglAddon = new WebglAddon();
|
|
|
|
webglAddon.onContextLoss(() => {
|
|
|
|
webglAddon.dispose();
|
|
|
|
});
|
|
|
|
this.terminal.loadAddon(webglAddon);
|
|
|
|
if (!loggedWebGL) {
|
|
|
|
console.log("loaded webgl!");
|
|
|
|
loggedWebGL = true;
|
|
|
|
}
|
|
|
|
}
|
2024-08-16 22:42:16 +02:00
|
|
|
this.terminal.parser.registerOscHandler(7, (data: string) => {
|
|
|
|
if (!this.loaded) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (data == null || data.length == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (data.startsWith("file://")) {
|
|
|
|
data = data.substring(7);
|
|
|
|
const nextSlashIdx = data.indexOf("/");
|
|
|
|
if (nextSlashIdx == -1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
data = data.substring(nextSlashIdx);
|
|
|
|
}
|
|
|
|
setTimeout(() => {
|
|
|
|
services.ObjectService.UpdateObjectMeta(WOS.makeORef("block", this.blockId), { "cmd:cwd": data });
|
|
|
|
}, 0);
|
|
|
|
return true;
|
|
|
|
});
|
2024-08-19 21:24:08 +02:00
|
|
|
this.terminal.attachCustomKeyEventHandler(waveOptions.keydownHandler);
|
2024-06-18 07:38:48 +02:00
|
|
|
this.connectElem = connectElem;
|
|
|
|
this.mainFileSubject = null;
|
|
|
|
this.heldData = [];
|
|
|
|
this.handleResize_debounced = debounce(50, this.handleResize.bind(this));
|
|
|
|
this.terminal.open(this.connectElem);
|
|
|
|
this.handleResize();
|
2024-06-24 23:34:31 +02:00
|
|
|
this.isRunning = true;
|
2024-06-18 07:38:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async initTerminal() {
|
2024-10-14 19:05:38 +02:00
|
|
|
const copyOnSelectAtom = getSettingsKeyAtom("term:copyonselect");
|
2024-06-18 07:38:48 +02:00
|
|
|
this.terminal.onData(this.handleTermData.bind(this));
|
2024-10-14 19:05:38 +02:00
|
|
|
this.terminal.onSelectionChange(
|
|
|
|
debounce(50, () => {
|
|
|
|
if (!globalStore.get(copyOnSelectAtom)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const selectedText = this.terminal.getSelection();
|
|
|
|
if (selectedText.length > 0) {
|
|
|
|
navigator.clipboard.writeText(selectedText);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
2024-08-13 00:53:34 +02:00
|
|
|
this.mainFileSubject = getFileSubject(this.blockId, TermFileName);
|
2024-06-18 07:38:48 +02:00
|
|
|
this.mainFileSubject.subscribe(this.handleNewFileSubjectData.bind(this));
|
|
|
|
try {
|
|
|
|
await this.loadInitialTerminalData();
|
|
|
|
} finally {
|
|
|
|
this.loaded = true;
|
|
|
|
}
|
|
|
|
this.runProcessIdleTimeout();
|
|
|
|
}
|
|
|
|
|
2024-06-24 23:34:31 +02:00
|
|
|
setIsRunning(isRunning: boolean) {
|
|
|
|
this.isRunning = isRunning;
|
|
|
|
}
|
|
|
|
|
2024-06-18 07:38:48 +02:00
|
|
|
dispose() {
|
|
|
|
this.terminal.dispose();
|
|
|
|
this.mainFileSubject.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
handleTermData(data: string) {
|
2024-08-19 20:54:54 +02:00
|
|
|
const b64data = util.stringToBase64(data);
|
2024-10-17 23:34:02 +02:00
|
|
|
RpcApi.ControllerInputCommand(TabRpcClient, { blockid: this.blockId, inputdata64: b64data });
|
2024-06-18 07:38:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
addFocusListener(focusFn: () => void) {
|
|
|
|
this.terminal.textarea.addEventListener("focus", focusFn);
|
|
|
|
}
|
|
|
|
|
|
|
|
handleNewFileSubjectData(msg: WSFileEventData) {
|
2024-06-24 23:34:31 +02:00
|
|
|
if (msg.fileop == "truncate") {
|
|
|
|
this.terminal.clear();
|
|
|
|
this.heldData = [];
|
|
|
|
} else if (msg.fileop == "append") {
|
|
|
|
const decodedData = base64ToArray(msg.data64);
|
|
|
|
if (this.loaded) {
|
|
|
|
this.doTerminalWrite(decodedData, null);
|
|
|
|
} else {
|
|
|
|
this.heldData.push(decodedData);
|
|
|
|
}
|
|
|
|
} else {
|
2024-06-18 07:38:48 +02:00
|
|
|
console.log("bad fileop for terminal", msg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
doTerminalWrite(data: string | Uint8Array, setPtyOffset?: number): Promise<void> {
|
|
|
|
let resolve: () => void = null;
|
|
|
|
let prtn = new Promise<void>((presolve, _) => {
|
|
|
|
resolve = presolve;
|
|
|
|
});
|
|
|
|
this.terminal.write(data, () => {
|
|
|
|
if (setPtyOffset != null) {
|
|
|
|
this.ptyOffset = setPtyOffset;
|
|
|
|
} else {
|
|
|
|
this.ptyOffset += data.length;
|
|
|
|
this.dataBytesProcessed += data.length;
|
|
|
|
}
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
return prtn;
|
|
|
|
}
|
|
|
|
|
|
|
|
async loadInitialTerminalData(): Promise<void> {
|
|
|
|
let startTs = Date.now();
|
2024-08-13 00:53:34 +02:00
|
|
|
const { data: cacheData, fileInfo: cacheFile } = await fetchWaveFile(this.blockId, TermCacheFileName);
|
2024-06-18 07:38:48 +02:00
|
|
|
let ptyOffset = 0;
|
|
|
|
if (cacheFile != null) {
|
|
|
|
ptyOffset = cacheFile.meta["ptyoffset"] ?? 0;
|
|
|
|
if (cacheData.byteLength > 0) {
|
2024-10-22 21:50:44 +02:00
|
|
|
const curTermSize: TermSize = { rows: this.terminal.rows, cols: this.terminal.cols };
|
|
|
|
const fileTermSize: TermSize = cacheFile.meta["termsize"];
|
|
|
|
let didResize = false;
|
|
|
|
if (
|
|
|
|
fileTermSize != null &&
|
|
|
|
(fileTermSize.rows != curTermSize.rows || fileTermSize.cols != curTermSize.cols)
|
|
|
|
) {
|
|
|
|
console.log("terminal restore size mismatch, temp resize", fileTermSize, curTermSize);
|
|
|
|
this.terminal.resize(fileTermSize.cols, fileTermSize.rows);
|
|
|
|
didResize = true;
|
|
|
|
}
|
2024-06-18 07:38:48 +02:00
|
|
|
this.doTerminalWrite(cacheData, ptyOffset);
|
2024-10-22 21:50:44 +02:00
|
|
|
if (didResize) {
|
|
|
|
this.terminal.resize(curTermSize.cols, curTermSize.rows);
|
|
|
|
}
|
2024-06-18 07:38:48 +02:00
|
|
|
}
|
|
|
|
}
|
2024-08-13 00:53:34 +02:00
|
|
|
const { data: mainData, fileInfo: mainFile } = await fetchWaveFile(this.blockId, TermFileName, ptyOffset);
|
2024-06-18 07:38:48 +02:00
|
|
|
console.log(
|
|
|
|
`terminal loaded cachefile:${cacheData?.byteLength ?? 0} main:${mainData?.byteLength ?? 0} bytes, ${Date.now() - startTs}ms`
|
|
|
|
);
|
|
|
|
if (mainFile != null) {
|
|
|
|
await this.doTerminalWrite(mainData, null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-05 09:21:08 +02:00
|
|
|
async resyncController(reason: string) {
|
|
|
|
dlog("resync controller", this.blockId, reason);
|
2024-10-17 23:34:02 +02:00
|
|
|
const tabId = globalStore.get(atoms.staticTabId);
|
2024-09-05 09:21:08 +02:00
|
|
|
const rtOpts: RuntimeOpts = { termsize: { rows: this.terminal.rows, cols: this.terminal.cols } };
|
|
|
|
try {
|
2024-10-17 23:34:02 +02:00
|
|
|
await RpcApi.ControllerResyncCommand(TabRpcClient, {
|
2024-09-16 20:59:39 +02:00
|
|
|
tabid: tabId,
|
|
|
|
blockid: this.blockId,
|
|
|
|
rtopts: rtOpts,
|
|
|
|
});
|
2024-09-05 09:21:08 +02:00
|
|
|
} catch (e) {
|
|
|
|
console.log(`error controller resync (${reason})`, this.blockId, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-18 07:38:48 +02:00
|
|
|
handleResize() {
|
|
|
|
const oldRows = this.terminal.rows;
|
|
|
|
const oldCols = this.terminal.cols;
|
|
|
|
this.fitAddon.fit();
|
|
|
|
if (oldRows !== this.terminal.rows || oldCols !== this.terminal.cols) {
|
2024-09-05 09:21:08 +02:00
|
|
|
const termSize: TermSize = { rows: this.terminal.rows, cols: this.terminal.cols };
|
2024-06-18 07:38:48 +02:00
|
|
|
const wsCommand: SetBlockTermSizeWSCommand = {
|
|
|
|
wscommand: "setblocktermsize",
|
|
|
|
blockid: this.blockId,
|
2024-09-05 09:21:08 +02:00
|
|
|
termsize: termSize,
|
2024-06-18 07:38:48 +02:00
|
|
|
};
|
|
|
|
sendWSCommand(wsCommand);
|
|
|
|
}
|
2024-09-05 09:21:08 +02:00
|
|
|
dlog("resize", `${this.terminal.rows}x${this.terminal.cols}`, `${oldRows}x${oldCols}`, this.hasResized);
|
|
|
|
if (!this.hasResized) {
|
|
|
|
this.hasResized = true;
|
|
|
|
this.resyncController("initial resize");
|
|
|
|
}
|
2024-06-18 07:38:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
processAndCacheData() {
|
2024-10-07 18:51:23 +02:00
|
|
|
if (this.dataBytesProcessed < MinDataProcessedForCache) {
|
2024-06-18 07:38:48 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const serializedOutput = this.serializeAddon.serialize();
|
2024-10-22 21:50:44 +02:00
|
|
|
const termSize: TermSize = { rows: this.terminal.rows, cols: this.terminal.cols };
|
|
|
|
console.log("idle timeout term", this.dataBytesProcessed, serializedOutput.length, termSize);
|
|
|
|
services.BlockService.SaveTerminalState(this.blockId, serializedOutput, "full", this.ptyOffset, termSize);
|
2024-06-18 07:38:48 +02:00
|
|
|
this.dataBytesProcessed = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
runProcessIdleTimeout() {
|
|
|
|
setTimeout(() => {
|
|
|
|
window.requestIdleCallback(() => {
|
|
|
|
this.processAndCacheData();
|
|
|
|
this.runProcessIdleTimeout();
|
|
|
|
});
|
|
|
|
}, 5000);
|
|
|
|
}
|
|
|
|
}
|