From 07ad5f063ea0554d5a16e69c8511ac4dc01b7722 Mon Sep 17 00:00:00 2001 From: Mike Sawka Date: Sun, 18 Feb 2024 17:58:22 -0800 Subject: [PATCH 1/4] Fix #302 and clean up extraneous terminal margins (#303) * font loading fix #302 * fix inconsistent paddings. issue was that the first '[' feels indented (because of the fixed with font). applying a negative 2px margin fixes it and allows us to remove the paddings on the other elements. --- src/app/line/lines.less | 10 ++++------ src/index.ts | 9 ++------- 2 files changed, 6 insertions(+), 13 deletions(-) diff --git a/src/app/line/lines.less b/src/app/line/lines.less index 520eee0aa..19c5bc62a 100644 --- a/src/app/line/lines.less +++ b/src/app/line/lines.less @@ -84,11 +84,14 @@ } .meta.meta-line1 { - margin-left: 2px; color: rgba(@base-color, 0.6) !important; font-size: 11px; } + .meta.meta-line2 { + margin-left: -2px; + } + &.has-rtnstate .terminal-wrapper { padding-bottom: 0; } @@ -113,11 +116,6 @@ overflow-x: hidden; } - .terminal { - margin-right: 8px; - padding: 0.25rem; - } - &.cmd-done .terminal .xterm-cursor { display: none; } diff --git a/src/index.ts b/src/index.ts index 7b415a4b0..87ebfcf2c 100644 --- a/src/index.ts +++ b/src/index.ts @@ -20,14 +20,9 @@ document.addEventListener("DOMContentLoaded", () => { let reactElem = React.createElement(App, null, null); let elem = document.getElementById("app"); let root = createRoot(elem); - let isFontLoaded = document.fonts.check("12px 'JetBrains Mono'"); - if (isFontLoaded) { + document.fonts.ready.then(() => { root.render(reactElem); - } else { - document.fonts.ready.then(() => { - root.render(reactElem); - }); - } + }); }); (window as any).mobx = mobx; From 7b6025fd4b97e8feff7c8001e130fd36f3008185 Mon Sep 17 00:00:00 2001 From: Mike Sawka Date: Tue, 20 Feb 2024 12:20:17 -0800 Subject: [PATCH 2/4] forward port v0.6.3 changes to main (#308) --- src/app/appconst.ts | 3 +- .../common/modals/viewremoteconndetail.tsx | 6 +- src/app/line/linecomps.tsx | 2 +- src/app/line/lines.less | 1 + src/app/magiclayout.ts | 2 - src/app/workspace/screen/screenview.tsx | 3 +- src/electron/emain.ts | 42 ++++++++++- src/electron/preload.js | 1 + src/index.ts | 2 + src/models/cmd.ts | 5 ++ src/models/historyview.ts | 4 +- src/models/model.ts | 69 +++++++++---------- src/models/model_old.ts-deprecated | 39 ++++++++++- src/models/screen.ts | 4 +- src/plugins/terminal/terminal.tsx | 3 +- src/types/custom.d.ts | 38 ++++++++++ src/util/textmeasure.ts | 34 +++++++-- 17 files changed, 197 insertions(+), 61 deletions(-) diff --git a/src/app/appconst.ts b/src/app/appconst.ts index 742331801..d8b05e742 100644 --- a/src/app/appconst.ts +++ b/src/app/appconst.ts @@ -18,7 +18,8 @@ export const ConfirmKey_HideShellPrompt = "hideshellprompt"; export const NoStrPos = -1; -export const RemotePtyRows = 8; // also in main.tsx +export const RemotePtyRows = 8; +export const RemotePtyTotalRows = 25; export const RemotePtyCols = 80; export const ProdServerEndpoint = "http://127.0.0.1:1619"; export const ProdServerWsEndpoint = "ws://127.0.0.1:1623"; diff --git a/src/app/common/modals/viewremoteconndetail.tsx b/src/app/common/modals/viewremoteconndetail.tsx index 48536c603..cc93e122c 100644 --- a/src/app/common/modals/viewremoteconndetail.tsx +++ b/src/app/common/modals/viewremoteconndetail.tsx @@ -370,7 +370,11 @@ class ViewRemoteConnDetailModal extends React.Component<{}, {}> { ref={this.termRef} data-remoteid={remote.remoteid} style={{ - height: textmeasure.termHeightFromRows(appconst.RemotePtyRows, termFontSize), + height: textmeasure.termHeightFromRows( + appconst.RemotePtyRows, + termFontSize, + appconst.RemotePtyTotalRows + ), width: termWidth, }} > diff --git a/src/app/line/linecomps.tsx b/src/app/line/linecomps.tsx index 306b03588..1b75962b6 100644 --- a/src/app/line/linecomps.tsx +++ b/src/app/line/linecomps.tsx @@ -388,7 +388,7 @@ class LineCmd extends React.Component< let height = 45 + 24; // height of zero height terminal const usedRows = screen.getUsedRows(lineutil.getRendererContext(line), line, cmd, width); if (usedRows > 0) { - height = 48 + 24 + termHeightFromRows(usedRows, GlobalModel.termFontSize.get()); + height = 48 + 24 + termHeightFromRows(usedRows, GlobalModel.termFontSize.get(), cmd.getTermMaxRows()); } return height; } diff --git a/src/app/line/lines.less b/src/app/line/lines.less index 19c5bc62a..6291d49e7 100644 --- a/src/app/line/lines.less +++ b/src/app/line/lines.less @@ -464,6 +464,7 @@ padding: 0 0 10px 0; flex-grow: 1; position: relative; + overflow-x: hidden; &::-webkit-scrollbar-thumb { background-color: transparent !important; diff --git a/src/app/magiclayout.ts b/src/app/magiclayout.ts index 4b2bb70f4..76d73dc92 100644 --- a/src/app/magiclayout.ts +++ b/src/app/magiclayout.ts @@ -18,8 +18,6 @@ let MagicLayout = { ScreenMinContentSize: 100, ScreenMaxContentSize: 5000, - // the 3 is for descenders, which get cut off in the terminal without this - TermDescendersHeight: 3, TermWidthBuffer: 15, TabWidth: 154, diff --git a/src/app/workspace/screen/screenview.tsx b/src/app/workspace/screen/screenview.tsx index 9e88fb2f2..f37dce701 100644 --- a/src/app/workspace/screen/screenview.tsx +++ b/src/app/workspace/screen/screenview.tsx @@ -109,6 +109,7 @@ class ScreenView extends React.Component<{ session: Session; screen: Screen }, { return
; } let fontSize = GlobalModel.termFontSize.get(); + let dprStr = sprintf("%0.3f", GlobalModel.devicePixelRatio.get()); let viewOpts = screen.viewOpts.get(); let hasSidebar = viewOpts?.sidebar?.open; let winWidth = "100%"; @@ -145,7 +146,7 @@ class ScreenView extends React.Component<{ session: Session; screen: Screen }, { return (
{ + if (MainWindow == null) { + return; + } + MainWindow.webContents.setZoomFactor(1); + MainWindow.webContents.send("zoom-changed"); + }, + }, + { + label: "Zoom In", + accelerator: cmdOrAlt + "+Plus", + click: () => { + if (MainWindow == null) { + return; + } + const zoomFactor = MainWindow.webContents.getZoomFactor(); + MainWindow.webContents.setZoomFactor(zoomFactor * 1.1); + MainWindow.webContents.send("zoom-changed"); + }, + }, + { + label: "Zoom Out", + accelerator: cmdOrAlt + "+-", + click: () => { + if (MainWindow == null) { + return; + } + const zoomFactor = MainWindow.webContents.getZoomFactor(); + MainWindow.webContents.setZoomFactor(zoomFactor / 1.1); + MainWindow.webContents.send("zoom-changed"); + }, + }, { type: "separator" }, { role: "togglefullscreen" }, ], @@ -375,6 +408,9 @@ function createMainWindow(clientData: ClientDataType | null) { win.on("close", () => { MainWindow = null; }); + win.webContents.on("zoom-changed", (e) => { + win.webContents.send("zoom-changed"); + }); win.webContents.setWindowOpenHandler(({ url, frameName }) => { if (url.startsWith("https://docs.waveterm.dev/")) { console.log("openExternal docs", url); diff --git a/src/electron/preload.js b/src/electron/preload.js index e7be2ec57..cd807c652 100644 --- a/src/electron/preload.js +++ b/src/electron/preload.js @@ -21,6 +21,7 @@ contextBridge.exposeInMainWorld("api", { onWCmd: (callback) => ipcRenderer.on("w-cmd", callback), onPCmd: (callback) => ipcRenderer.on("p-cmd", callback), onRCmd: (callback) => ipcRenderer.on("r-cmd", callback), + onZoomChanged: (callback) => ipcRenderer.on("zoom-changed", callback), onMetaArrowUp: (callback) => ipcRenderer.on("meta-arrowup", callback), onMetaArrowDown: (callback) => ipcRenderer.on("meta-arrowdown", callback), onMetaPageUp: (callback) => ipcRenderer.on("meta-pageup", callback), diff --git a/src/index.ts b/src/index.ts index 87ebfcf2c..2a2c699b5 100644 --- a/src/index.ts +++ b/src/index.ts @@ -8,6 +8,7 @@ import { sprintf } from "sprintf-js"; import { App } from "@/app/app"; import * as DOMPurify from "dompurify"; import { loadFonts } from "./util/util"; +import * as textmeasure from "./util/textmeasure"; // @ts-ignore let VERSION = __WAVETERM_VERSION__; @@ -28,5 +29,6 @@ document.addEventListener("DOMContentLoaded", () => { (window as any).mobx = mobx; (window as any).sprintf = sprintf; (window as any).DOMPurify = DOMPurify; +(window as any).textmeasure = textmeasure; console.log("WaveTerm", VERSION, BUILD); diff --git a/src/models/cmd.ts b/src/models/cmd.ts index a55eb0749..d2ab3aa85 100644 --- a/src/models/cmd.ts +++ b/src/models/cmd.ts @@ -88,6 +88,11 @@ class Cmd { return this.data.get().termopts; } + getTermMaxRows(): number { + let termOpts = this.getTermOpts(); + return termOpts?.rows; + } + getCmdStr(): string { return this.data.get().cmdstr; } diff --git a/src/models/historyview.ts b/src/models/historyview.ts index 44c0649a0..b309ac9ed 100644 --- a/src/models/historyview.ts +++ b/src/models/historyview.ts @@ -115,7 +115,7 @@ class HistoryViewModel { } else { this.activeItem.set(hitem.historyid); let width = termWidthFromCols(80, this.globalModel.termFontSize.get()); - let height = termHeightFromRows(25, this.globalModel.termFontSize.get()); + let height = termHeightFromRows(25, this.globalModel.termFontSize.get(), 25); this.specialLineContainer = new SpecialLineContainer( this, { width, height }, @@ -149,7 +149,7 @@ class HistoryViewModel { } _deleteSelected(): void { - let lineIds = Array.from(this.selectedItems.keys()); + let lineIds: string[] = Array.from(this.selectedItems.keys()); let prtn = GlobalCommandRunner.historyPurgeLines(lineIds); prtn.then((result: CommandRtnType) => { if (!result.success) { diff --git a/src/models/model.ts b/src/models/model.ts index d3f5841d9..f2f9dc980 100644 --- a/src/models/model.ts +++ b/src/models/model.ts @@ -32,13 +32,8 @@ import { MainSidebarModel } from "./mainsidebar"; import { Screen } from "./screen"; import { Cmd } from "./cmd"; import { GlobalCommandRunner } from "./global"; - -type KeyModsType = { - meta?: boolean; - ctrl?: boolean; - alt?: boolean; - shift?: boolean; -}; +import { clearMonoFontCache } from "@/util/textmeasure"; +import type { TermWrap } from "@/plugins/terminal/term"; type SWLinePtr = { line: LineType; @@ -46,36 +41,6 @@ type SWLinePtr = { screen: Screen; }; -type ElectronApi = { - getId: () => string; - getIsDev: () => boolean; - getPlatform: () => string; - getAuthKey: () => string; - getWaveSrvStatus: () => boolean; - restartWaveSrv: () => boolean; - reloadWindow: () => void; - openExternalLink: (url: string) => void; - reregisterGlobalShortcut: (shortcut: string) => void; - onTCmd: (callback: (mods: KeyModsType) => void) => void; - onICmd: (callback: (mods: KeyModsType) => void) => void; - onLCmd: (callback: (mods: KeyModsType) => void) => void; - onHCmd: (callback: (mods: KeyModsType) => void) => void; - onPCmd: (callback: (mods: KeyModsType) => void) => void; - onRCmd: (callback: (mods: KeyModsType) => void) => void; - onWCmd: (callback: (mods: KeyModsType) => void) => void; - onMenuItemAbout: (callback: () => void) => void; - onMetaArrowUp: (callback: () => void) => void; - onMetaArrowDown: (callback: () => void) => void; - onMetaPageUp: (callback: () => void) => void; - onMetaPageDown: (callback: () => void) => void; - onBracketCmd: (callback: (event: any, arg: { relative: number }, mods: KeyModsType) => void) => void; - onDigitCmd: (callback: (event: any, arg: { digit: number }, mods: KeyModsType) => void) => void; - contextScreen: (screenOpts: { screenId: string }, position: { x: number; y: number }) => void; - contextEditMenu: (position: { x: number; y: number }, opts: ContextMenuOpts) => void; - onWaveSrvStatusChange: (callback: (status: boolean, pid: number) => void) => void; - getLastLogs: (numOfLines: number, callback: (logs: any) => void) => void; -}; - function getApi(): ElectronApi { return (window as any).api; } @@ -133,7 +98,10 @@ class Model { }); lineSettingsModal: OV = mobx.observable.box(null, { name: "lineSettingsModal", - }); // linenum + }); + devicePixelRatio: OV = mobx.observable.box(window.devicePixelRatio, { + name: "devicePixelRatio", + }); remotesModel: RemotesModel; inputModel: InputModel; @@ -196,6 +164,7 @@ class Model { getApi().onPCmd(this.onPCmd.bind(this)); getApi().onWCmd(this.onWCmd.bind(this)); getApi().onRCmd(this.onRCmd.bind(this)); + getApi().onZoomChanged(this.onZoomChanged.bind(this)); getApi().onMenuItemAbout(this.onMenuItemAbout.bind(this)); getApi().onMetaArrowUp(this.onMetaArrowUp.bind(this)); getApi().onMetaArrowDown(this.onMetaArrowDown.bind(this)); @@ -514,6 +483,30 @@ class Model { } } + onZoomChanged(): void { + mobx.action(() => { + this.devicePixelRatio.set(window.devicePixelRatio); + clearMonoFontCache(); + })(); + } + + // for debuggin + getSelectedTermWrap(): TermWrap { + let screen = this.getActiveScreen(); + if (screen == null) { + return null; + } + let lineNum = screen.selectedLine.get(); + if (lineNum == null) { + return null; + } + let line = screen.getLineByNum(lineNum); + if (line == null) { + return null; + } + return screen.getTermWrap(line.lineid); + } + clearModals(): boolean { let didSomething = false; mobx.action(() => { diff --git a/src/models/model_old.ts-deprecated b/src/models/model_old.ts-deprecated index 5368a8937..d0cc731bb 100644 --- a/src/models/model_old.ts-deprecated +++ b/src/models/model_old.ts-deprecated @@ -205,6 +205,7 @@ type ElectronApi = { onPCmd: (callback: (mods: KeyModsType) => void) => void; onRCmd: (callback: (mods: KeyModsType) => void) => void; onWCmd: (callback: (mods: KeyModsType) => void) => void; + onZoomChanged: (callback: () => void) => void; onMenuItemAbout: (callback: () => void) => void; onMetaArrowUp: (callback: () => void) => void; onMetaArrowDown: (callback: () => void) => void; @@ -308,6 +309,11 @@ class Cmd { return this.data.get().termopts; } + getTermMaxRows(): number { + let termOpts = this.getTermOpts(); + return termOpts?.rows; + } + getCmdStr(): string { return this.data.get().cmdstr; } @@ -743,7 +749,7 @@ class Screen { getMaxContentSize(): WindowSize { if (this.lastScreenSize == null) { let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get()); + let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); return { width, height }; } let winSize = this.lastScreenSize; @@ -757,7 +763,7 @@ class Screen { getIdealContentSize(): WindowSize { if (this.lastScreenSize == null) { let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get()); + let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); return { width, height }; } let winSize = this.lastScreenSize; @@ -2396,7 +2402,7 @@ class HistoryViewModel { } else { this.activeItem.set(hitem.historyid); let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get()); + let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); this.specialLineContainer = new SpecialLineContainer( this, { width, height }, @@ -3451,6 +3457,9 @@ class Model { lineSettingsModal: OV = mobx.observable.box(null, { name: "lineSettingsModal", }); // linenum + devicePixelRatio: OV = mobx.observable.box(window.devicePixelRatio, { + name: "devicePixelRatio", + }); remotesModalModel: RemotesModalModel; remotesModel: RemotesModel; @@ -3515,6 +3524,7 @@ class Model { getApi().onPCmd(this.onPCmd.bind(this)); getApi().onWCmd(this.onWCmd.bind(this)); getApi().onRCmd(this.onRCmd.bind(this)); + getApi().onZoomChanged(this.onZoomChanged.bind(this)); getApi().onMenuItemAbout(this.onMenuItemAbout.bind(this)); getApi().onMetaArrowUp(this.onMetaArrowUp.bind(this)); getApi().onMetaArrowDown(this.onMetaArrowDown.bind(this)); @@ -3826,6 +3836,29 @@ class Model { } } + onZoomChanged(): void { + mobx.action(() => { + this.devicePixelRatio.set(window.devicePixelRatio); + clearMonoFontCache(); + })(); + } + + getSelectedTermWrap(): TermWrap { + let screen = this.getActiveScreen(); + if (screen == null) { + return null; + } + let lineNum = screen.selectedLine.get(); + if (lineNum == null) { + return null; + } + let line = screen.getLineByNum(lineNum); + if (line == null) { + return null; + } + return screen.getTermWrap(line.lineid); + } + clearModals(): boolean { let didSomething = false; mobx.action(() => { diff --git a/src/models/screen.ts b/src/models/screen.ts index 9b0366711..624554711 100644 --- a/src/models/screen.ts +++ b/src/models/screen.ts @@ -410,7 +410,7 @@ class Screen { getMaxContentSize(): WindowSize { if (this.lastScreenSize == null) { let width = termWidthFromCols(80, this.globalModel.termFontSize.get()); - let height = termHeightFromRows(25, this.globalModel.termFontSize.get()); + let height = termHeightFromRows(25, this.globalModel.termFontSize.get(), 25); return { width, height }; } let winSize = this.lastScreenSize; @@ -424,7 +424,7 @@ class Screen { getIdealContentSize(): WindowSize { if (this.lastScreenSize == null) { let width = termWidthFromCols(80, this.globalModel.termFontSize.get()); - let height = termHeightFromRows(25, this.globalModel.termFontSize.get()); + let height = termHeightFromRows(25, this.globalModel.termFontSize.get(), 25); return { width, height }; } let winSize = this.lastScreenSize; diff --git a/src/plugins/terminal/terminal.tsx b/src/plugins/terminal/terminal.tsx index 261056265..31bd33f1e 100644 --- a/src/plugins/terminal/terminal.tsx +++ b/src/plugins/terminal/terminal.tsx @@ -148,7 +148,7 @@ class TerminalRenderer extends React.Component< .get(); let cmd = screen.getCmd(line); // will not be null let usedRows = screen.getUsedRows(lineutil.getRendererContext(line), line, cmd, width); - let termHeight = termHeightFromRows(usedRows, GlobalModel.termFontSize.get()); + let termHeight = termHeightFromRows(usedRows, GlobalModel.termFontSize.get(), cmd.getTermMaxRows()); if (usedRows === 0) { termHeight = 0; } @@ -164,6 +164,7 @@ class TerminalRenderer extends React.Component< { "zero-height": termHeight == 0 }, { collapsed: collapsed } )} + data-usedrows={usedRows} >
diff --git a/src/types/custom.d.ts b/src/types/custom.d.ts index 500746c2a..f93f76423 100644 --- a/src/types/custom.d.ts +++ b/src/types/custom.d.ts @@ -833,6 +833,44 @@ declare global { isLineIdInSidebar(lineId: string): boolean; getContainerType(): LineContainerStrs; }; + + type KeyModsType = { + meta?: boolean; + ctrl?: boolean; + alt?: boolean; + shift?: boolean; + }; + + type ElectronApi = { + getId: () => string; + getIsDev: () => boolean; + getPlatform: () => string; + getAuthKey: () => string; + getWaveSrvStatus: () => boolean; + restartWaveSrv: () => boolean; + reloadWindow: () => void; + openExternalLink: (url: string) => void; + reregisterGlobalShortcut: (shortcut: string) => void; + onTCmd: (callback: (mods: KeyModsType) => void) => void; + onICmd: (callback: (mods: KeyModsType) => void) => void; + onLCmd: (callback: (mods: KeyModsType) => void) => void; + onHCmd: (callback: (mods: KeyModsType) => void) => void; + onPCmd: (callback: (mods: KeyModsType) => void) => void; + onRCmd: (callback: (mods: KeyModsType) => void) => void; + onWCmd: (callback: (mods: KeyModsType) => void) => void; + onZoomChanged: (callback: () => void) => void; + onMenuItemAbout: (callback: () => void) => void; + onMetaArrowUp: (callback: () => void) => void; + onMetaArrowDown: (callback: () => void) => void; + onMetaPageUp: (callback: () => void) => void; + onMetaPageDown: (callback: () => void) => void; + onBracketCmd: (callback: (event: any, arg: { relative: number }, mods: KeyModsType) => void) => void; + onDigitCmd: (callback: (event: any, arg: { digit: number }, mods: KeyModsType) => void) => void; + contextScreen: (screenOpts: { screenId: string }, position: { x: number; y: number }) => void; + contextEditMenu: (position: { x: number; y: number }, opts: ContextMenuOpts) => void; + onWaveSrvStatusChange: (callback: (status: boolean, pid: number) => void) => void; + getLastLogs: (numOfLines: number, callback: (logs: any) => void) => void; + }; } export {}; diff --git a/src/util/textmeasure.ts b/src/util/textmeasure.ts index 9551e00bd..5f3f5ca91 100644 --- a/src/util/textmeasure.ts +++ b/src/util/textmeasure.ts @@ -30,6 +30,10 @@ function getMonoFontSize(fontSize: number): { height: number; width: number } { return size; } +function clearMonoFontCache(): void { + MonoFontSizes = []; +} + function measureText( text: string, textOpts?: { pre?: boolean; mono?: boolean; fontSize?: number | string } @@ -57,8 +61,9 @@ function measureText( throw new Error("cannot measure text, no #measure div"); } measureDiv.replaceChildren(textElem); - let rect = textElem.getBoundingClientRect(); - return { width: rect.width, height: Math.ceil(rect.height) }; + let height = textElem.offsetHeight; + let width = textElem.offsetWidth; + return { width: width, height: Math.ceil(height) }; } function windowWidthToCols(width: number, fontSize: number): number { @@ -82,10 +87,27 @@ function termWidthFromCols(cols: number, fontSize: number): number { return Math.ceil(dr.width * cols) + MagicLayout.TermWidthBuffer; } -function termHeightFromRows(rows: number, fontSize: number): number { +// we need to match the xtermjs calculation in CharSizeService.ts and DomRenderer.ts +// it does some crazy rounding depending on the value of window.devicePixelRatio +// works out to `realHeight = round(ceil(height * dpr) * rows / dpr) / rows` +// their calculation is based off the "totalRows" (so that argument has been added) +function termHeightFromRows(rows: number, fontSize: number, totalRows: number): number { let dr = getMonoFontSize(fontSize); - // TODO: replace the TermDescendersHeight with some calculation based on termFontSize. - return Math.ceil(dr.height * rows) + MagicLayout.TermDescendersHeight; + const dpr = window.devicePixelRatio; + if (totalRows == null || totalRows == 0) { + totalRows = rows > 25 ? rows : 25; + } + let realHeight = Math.round((Math.ceil(dr.height * dpr) * totalRows) / dpr) / totalRows; + return Math.ceil(realHeight * rows); } -export { measureText, getMonoFontSize, windowWidthToCols, windowHeightToRows, termWidthFromCols, termHeightFromRows }; +export { + measureText, + getMonoFontSize, + windowWidthToCols, + windowHeightToRows, + termWidthFromCols, + termHeightFromRows, + clearMonoFontCache, + MonoFontSizes, +}; From 8cb449d2e72c88ac462309de6048cc5bc51980ff Mon Sep 17 00:00:00 2001 From: Red J Adaya Date: Wed, 21 Feb 2024 04:22:09 +0800 Subject: [PATCH 3/4] remove old model (#307) --- src/models/model_old.ts-deprecated | 5248 ---------------------------- 1 file changed, 5248 deletions(-) delete mode 100644 src/models/model_old.ts-deprecated diff --git a/src/models/model_old.ts-deprecated b/src/models/model_old.ts-deprecated deleted file mode 100644 index d0cc731bb..000000000 --- a/src/models/model_old.ts-deprecated +++ /dev/null @@ -1,5248 +0,0 @@ -// Copyright 2023, Command Line Inc. -// SPDX-License-Identifier: Apache-2.0 - -import type React from "react"; -import * as mobx from "mobx"; -import { sprintf } from "sprintf-js"; -import { v4 as uuidv4 } from "uuid"; -import { boundMethod } from "autobind-decorator"; -import { debounce } from "throttle-debounce"; -import * as mobxReact from "mobx-react"; -import { - handleJsonFetchResponse, - base64ToString, - stringToBase64, - base64ToArray, - genMergeData, - genMergeDataMap, - genMergeSimpleData, - boundInt, - isModKeyPress, -} from "@/util/util"; -import { TermWrap } from "@/plugins/terminal/term"; -import { PluginModel } from "@/plugins/plugins"; -import { - SessionDataType, - LineType, - RemoteType, - HistoryItem, - RemoteInstanceType, - RemotePtrType, - CmdDataType, - FeCmdPacketType, - TermOptsType, - ScreenDataType, - ScreenOptsType, - PtyDataUpdateType, - ModelUpdateType, - UpdateMessage, - InfoType, - UIContextType, - HistoryInfoType, - HistoryQueryOpts, - FeInputPacketType, - RemoteInputPacketType, - ContextMenuOpts, - RendererContext, - RendererModel, - PtyDataType, - BookmarkType, - ClientDataType, - HistoryViewDataType, - AlertMessageType, - HistorySearchParams, - UserInputRequest, - UserInputResponsePacket, - FocusTypeStrs, - ScreenLinesType, - HistoryTypeStrs, - RendererPluginType, - WindowSize, - WebShareOpts, - TermContextUnion, - RemoteEditType, - RemoteViewType, - CommandRtnType, - WebCmd, - WebRemote, - OpenAICmdInfoChatMessageType, - StatusIndicatorLevel, -} from "../types/types"; -import * as T from "../types/types"; -import { WSControl } from "./ws"; -import { - getMonoFontSize, - windowWidthToCols, - windowHeightToRows, - termWidthFromCols, - termHeightFromRows, -} from "@/util/textmeasure"; -import dayjs from "dayjs"; -import localizedFormat from "dayjs/plugin/localizedFormat"; -import customParseFormat from "dayjs/plugin/customParseFormat"; -import { getRendererContext, cmdStatusIsRunning } from "@/app/line/lineutil"; -import { MagicLayout } from "@/app/magiclayout"; -import { modalsRegistry } from "@/modals/registry"; -import * as appconst from "@/app/appconst"; -import { checkKeyPressed, adaptFromReactOrNativeKeyEvent, setKeyUtilPlatform } from "@/util/keyutil"; - -dayjs.extend(customParseFormat); -dayjs.extend(localizedFormat); - -const RemotePtyRows = 8; // also in main.tsx -const RemotePtyCols = 80; -const ProdServerEndpoint = "http://127.0.0.1:1619"; -const ProdServerWsEndpoint = "ws://127.0.0.1:1623"; -const DevServerEndpoint = "http://127.0.0.1:8090"; -const DevServerWsEndpoint = "ws://127.0.0.1:8091"; -const DefaultTermFontSize = 12; -const MinFontSize = 8; -const MaxFontSize = 24; -const InputChunkSize = 500; -const RemoteColors = ["red", "green", "yellow", "blue", "magenta", "cyan", "white", "orange"]; -const TabColors = ["red", "orange", "yellow", "green", "mint", "cyan", "blue", "violet", "pink", "white"]; -const TabIcons = [ - "sparkle", - "fire", - "ghost", - "cloud", - "compass", - "crown", - "droplet", - "graduation-cap", - "heart", - "file", -]; - -// @ts-ignore -const VERSION = __WAVETERM_VERSION__; -// @ts-ignore -const BUILD = __WAVETERM_BUILD__; - -type LineContainerModel = { - loadTerminalRenderer: (elem: Element, line: LineType, cmd: Cmd, width: number) => void; - registerRenderer: (lineId: string, renderer: RendererModel) => void; - unloadRenderer: (lineId: string) => void; - getIsFocused: (lineNum: number) => boolean; - getTermWrap: (lineId: string) => TermWrap; - getRenderer: (lineId: string) => RendererModel; - getFocusType: () => FocusTypeStrs; - getSelectedLine: () => number; - getCmd: (line: LineType) => Cmd; - setLineFocus: (lineNum: number, focus: boolean) => void; - getUsedRows: (context: RendererContext, line: LineType, cmd: Cmd, width: number) => number; - getContentHeight: (context: RendererContext) => number; - setContentHeight: (context: RendererContext, height: number) => void; - getMaxContentSize(): WindowSize; - getIdealContentSize(): WindowSize; - isSidebarOpen(): boolean; - isLineIdInSidebar(lineId: string): boolean; - getContainerType(): T.LineContainerStrs; -}; - -type SWLinePtr = { - line: LineType; - slines: ScreenLines; - screen: Screen; -}; - -function keyHasNoMods(e: any) { - return !e.shiftKey && !e.ctrlKey && !e.metaKey && !e.altKey; -} - -type OV = mobx.IObservableValue; -type OArr = mobx.IObservableArray; -type OMap = mobx.ObservableMap; -type CV = mobx.IComputedValue; - -function isBlank(s: string) { - return s == null || s == ""; -} - -function remotePtrToString(rptr: RemotePtrType): string { - if (rptr == null || isBlank(rptr.remoteid)) { - return null; - } - if (isBlank(rptr.ownerid) && isBlank(rptr.name)) { - return rptr.remoteid; - } - if (!isBlank(rptr.ownerid) && isBlank(rptr.name)) { - return sprintf("@%s:%s", rptr.ownerid, rptr.remoteid); - } - if (isBlank(rptr.ownerid) && !isBlank(rptr.name)) { - return sprintf("%s:%s", rptr.remoteid, rptr.name); - } - return sprintf("@%s:%s:%s", rptr.ownerid, rptr.remoteid, rptr.name); -} - -function riToRPtr(ri: RemoteInstanceType): RemotePtrType { - if (ri == null) { - return null; - } - return { ownerid: ri.remoteownerid, remoteid: ri.remoteid, name: ri.name }; -} - -type KeyModsType = { - meta?: boolean; - ctrl?: boolean; - alt?: boolean; - shift?: boolean; -}; - -type ElectronApi = { - getId: () => string; - getIsDev: () => boolean; - getPlatform: () => string; - getAuthKey: () => string; - getWaveSrvStatus: () => boolean; - restartWaveSrv: () => boolean; - reloadWindow: () => void; - openExternalLink: (url: string) => void; - onTCmd: (callback: (mods: KeyModsType) => void) => void; - onICmd: (callback: (mods: KeyModsType) => void) => void; - onLCmd: (callback: (mods: KeyModsType) => void) => void; - onHCmd: (callback: (mods: KeyModsType) => void) => void; - onPCmd: (callback: (mods: KeyModsType) => void) => void; - onRCmd: (callback: (mods: KeyModsType) => void) => void; - onWCmd: (callback: (mods: KeyModsType) => void) => void; - onZoomChanged: (callback: () => void) => void; - onMenuItemAbout: (callback: () => void) => void; - onMetaArrowUp: (callback: () => void) => void; - onMetaArrowDown: (callback: () => void) => void; - onMetaPageUp: (callback: () => void) => void; - onMetaPageDown: (callback: () => void) => void; - onBracketCmd: (callback: (event: any, arg: { relative: number }, mods: KeyModsType) => void) => void; - onDigitCmd: (callback: (event: any, arg: { digit: number }, mods: KeyModsType) => void) => void; - contextScreen: (screenOpts: { screenId: string }, position: { x: number; y: number }) => void; - contextEditMenu: (position: { x: number; y: number }, opts: ContextMenuOpts) => void; - onWaveSrvStatusChange: (callback: (status: boolean, pid: number) => void) => void; - getLastLogs: (numOfLines: number, callback: (logs: any) => void) => void; -}; - -function getApi(): ElectronApi { - return (window as any).api; -} - -// clean empty string -function ces(s: string) { - if (s == "") { - return null; - } - return s; -} - -class Cmd { - screenId: string; - remote: RemotePtrType; - lineId: string; - data: OV; - - constructor(cmd: CmdDataType) { - this.screenId = cmd.screenid; - this.lineId = cmd.lineid; - this.remote = cmd.remote; - this.data = mobx.observable.box(cmd, { deep: false, name: "cmd-data" }); - } - - setCmd(cmd: CmdDataType) { - mobx.action(() => { - let origData = this.data.get(); - this.data.set(cmd); - if (origData != null && cmd != null && origData.status != cmd.status) { - GlobalModel.cmdStatusUpdate(this.screenId, this.lineId, origData.status, cmd.status); - } - })(); - } - - getRestartTs(): number { - return this.data.get().restartts; - } - - getAsWebCmd(lineid: string): WebCmd { - let cmd = this.data.get(); - let remote = GlobalModel.getRemote(this.remote.remoteid); - let webRemote: WebRemote = null; - if (remote != null) { - webRemote = { - remoteid: cmd.remote.remoteid, - alias: remote.remotealias, - canonicalname: remote.remotecanonicalname, - name: this.remote.name, - homedir: remote.remotevars["home"], - isroot: !!remote.remotevars["isroot"], - }; - } - let webCmd: WebCmd = { - screenid: cmd.screenid, - lineid: lineid, - remote: webRemote, - status: cmd.status, - cmdstr: cmd.cmdstr, - rawcmdstr: cmd.rawcmdstr, - festate: cmd.festate, - termopts: cmd.termopts, - cmdpid: cmd.cmdpid, - remotepid: cmd.remotepid, - donets: cmd.donets, - exitcode: cmd.exitcode, - durationms: cmd.durationms, - rtnstate: cmd.rtnstate, - vts: 0, - rtnstatestr: null, - }; - return webCmd; - } - - getExitCode(): number { - return this.data.get().exitcode; - } - - getRtnState(): boolean { - return this.data.get().rtnstate; - } - - getStatus(): string { - return this.data.get().status; - } - - getTermOpts(): TermOptsType { - return this.data.get().termopts; - } - - getTermMaxRows(): number { - let termOpts = this.getTermOpts(); - return termOpts?.rows; - } - - getCmdStr(): string { - return this.data.get().cmdstr; - } - - getRemoteFeState(): Record { - return this.data.get().festate; - } - - isRunning(): boolean { - let data = this.data.get(); - return cmdStatusIsRunning(data.status); - } - - handleData(data: string, termWrap: TermWrap): void { - if (!this.isRunning()) { - return; - } - for (let pos = 0; pos < data.length; pos += InputChunkSize) { - let dataChunk = data.slice(pos, pos + InputChunkSize); - this.handleInputChunk(dataChunk); - } - } - - handleDataFromRenderer(data: string, renderer: RendererModel): void { - if (!this.isRunning()) { - return; - } - for (let pos = 0; pos < data.length; pos += InputChunkSize) { - let dataChunk = data.slice(pos, pos + InputChunkSize); - this.handleInputChunk(dataChunk); - } - } - - handleInputChunk(data: string): void { - let inputPacket: FeInputPacketType = { - type: "feinput", - ck: this.screenId + "/" + this.lineId, - remote: this.remote, - inputdata64: stringToBase64(data), - }; - GlobalModel.sendInputPacket(inputPacket); - } -} - -class Screen { - sessionId: string; - screenId: string; - screenIdx: OV; - opts: OV; - viewOpts: OV; - name: OV; - archived: OV; - curRemote: OV; - nextLineNum: OV; - lastScreenSize: WindowSize; - lastCols: number; - lastRows: number; - selectedLine: OV; - focusType: OV; - anchor: OV<{ anchorLine: number; anchorOffset: number }>; - termLineNumFocus: OV; - setAnchor_debounced: (anchorLine: number, anchorOffset: number) => void; - terminals: Record = {}; // lineid => TermWrap - renderers: Record = {}; // lineid => RendererModel - shareMode: OV; - webShareOpts: OV; - filterRunning: OV; - statusIndicator: OV; - numRunningCmds: OV; - - constructor(sdata: ScreenDataType) { - this.sessionId = sdata.sessionid; - this.screenId = sdata.screenid; - this.name = mobx.observable.box(sdata.name, { name: "screen-name" }); - this.nextLineNum = mobx.observable.box(sdata.nextlinenum, { name: "screen-nextlinenum" }); - this.screenIdx = mobx.observable.box(sdata.screenidx, { - name: "screen-screenidx", - }); - this.opts = mobx.observable.box(sdata.screenopts, { name: "screen-opts" }); - this.viewOpts = mobx.observable.box(sdata.screenviewopts, { name: "viewOpts" }); - this.archived = mobx.observable.box(!!sdata.archived, { - name: "screen-archived", - }); - this.focusType = mobx.observable.box(sdata.focustype, { - name: "focusType", - }); - this.selectedLine = mobx.observable.box(sdata.selectedline == 0 ? null : sdata.selectedline, { - name: "selectedLine", - }); - this.setAnchor_debounced = debounce(1000, this.setAnchor.bind(this)); - this.anchor = mobx.observable.box( - { anchorLine: sdata.selectedline, anchorOffset: 0 }, - { name: "screen-anchor" } - ); - this.termLineNumFocus = mobx.observable.box(0, { - name: "termLineNumFocus", - }); - this.curRemote = mobx.observable.box(sdata.curremote, { - name: "screen-curRemote", - }); - this.shareMode = mobx.observable.box(sdata.sharemode, { - name: "screen-shareMode", - }); - this.webShareOpts = mobx.observable.box(sdata.webshareopts, { - name: "screen-webShareOpts", - }); - this.filterRunning = mobx.observable.box(false, { - name: "screen-filter-running", - }); - this.statusIndicator = mobx.observable.box(StatusIndicatorLevel.None, { - name: "screen-status-indicator", - }); - this.numRunningCmds = mobx.observable.box(0, { - name: "screen-num-running-cmds", - }); - } - - dispose() {} - - isWebShared(): boolean { - return this.shareMode.get() == "web" && this.webShareOpts.get() != null; - } - - isSidebarOpen(): boolean { - let viewOpts = this.viewOpts.get(); - if (viewOpts == null) { - return false; - } - return viewOpts.sidebar?.open; - } - - isLineIdInSidebar(lineId: string): boolean { - let viewOpts = this.viewOpts.get(); - if (viewOpts == null) { - return false; - } - if (!viewOpts.sidebar?.open) { - return false; - } - return viewOpts?.sidebar?.sidebarlineid == lineId; - } - - getContainerType(): T.LineContainerStrs { - return appconst.LineContainer_Main; - } - - getShareName(): string { - if (!this.isWebShared()) { - return null; - } - let opts = this.webShareOpts.get(); - if (opts == null) { - return null; - } - return opts.sharename; - } - - getWebShareUrl(): string { - let viewKey: string = null; - if (this.webShareOpts.get() != null) { - viewKey = this.webShareOpts.get().viewkey; - } - if (viewKey == null) { - return null; - } - if (GlobalModel.isDev) { - return sprintf( - "http://devtest.getprompt.com:9001/static/index-dev.html?screenid=%s&viewkey=%s", - this.screenId, - viewKey - ); - } - return sprintf("https://share.getprompt.dev/share/%s?viewkey=%s", this.screenId, viewKey); - } - - mergeData(data: ScreenDataType) { - if (data.sessionid != this.sessionId || data.screenid != this.screenId) { - throw new Error("invalid screen update, ids don't match"); - } - mobx.action(() => { - this.screenIdx.set(data.screenidx); - this.opts.set(data.screenopts); - this.viewOpts.set(data.screenviewopts); - this.name.set(data.name); - this.nextLineNum.set(data.nextlinenum); - this.archived.set(!!data.archived); - let oldSelectedLine = this.selectedLine.get(); - let oldFocusType = this.focusType.get(); - this.selectedLine.set(data.selectedline); - this.curRemote.set(data.curremote); - this.focusType.set(data.focustype); - this.refocusLine(data, oldFocusType, oldSelectedLine); - this.shareMode.set(data.sharemode); - this.webShareOpts.set(data.webshareopts); - // do not update anchorLine/anchorOffset (only stored) - })(); - } - - getContentHeight(context: RendererContext): number { - return GlobalModel.getContentHeight(context); - } - - setContentHeight(context: RendererContext, height: number): void { - GlobalModel.setContentHeight(context, height); - } - - getCmd(line: LineType): Cmd { - return GlobalModel.getCmd(line); - } - - getCmdById(lineId: string): Cmd { - return GlobalModel.getCmdByScreenLine(this.screenId, lineId); - } - - getAnchorStr(): string { - let anchor = this.anchor.get(); - if (anchor.anchorLine == null || anchor.anchorLine == 0) { - return "0"; - } - return sprintf("%d:%d", anchor.anchorLine, anchor.anchorOffset); - } - - getTabColor(): string { - let tabColor = "default"; - let screenOpts = this.opts.get(); - if (screenOpts != null && !isBlank(screenOpts.tabcolor)) { - tabColor = screenOpts.tabcolor; - } - return tabColor; - } - - getTabIcon(): string { - let tabIcon = "default"; - let screenOpts = this.opts.get(); - if (screenOpts != null && !isBlank(screenOpts.tabicon)) { - tabIcon = screenOpts.tabicon; - } - return tabIcon; - } - - getCurRemoteInstance(): RemoteInstanceType { - let session = GlobalModel.getSessionById(this.sessionId); - let rptr = this.curRemote.get(); - if (rptr == null) { - return null; - } - return session.getRemoteInstance(this.screenId, rptr); - } - - setAnchorFields(anchorLine: number, anchorOffset: number, reason: string): void { - mobx.action(() => { - this.anchor.set({ anchorLine: anchorLine, anchorOffset: anchorOffset }); - })(); - } - - refocusLine(sdata: ScreenDataType, oldFocusType: string, oldSelectedLine: number): void { - let isCmdFocus = sdata.focustype == "cmd"; - if (!isCmdFocus) { - return; - } - let curLineFocus = GlobalModel.getFocusedLine(); - let sline: LineType = null; - if (sdata.selectedline != 0) { - sline = this.getLineByNum(sdata.selectedline); - } - if ( - curLineFocus.cmdInputFocus || - (curLineFocus.linenum != null && curLineFocus.linenum != sdata.selectedline) - ) { - (document.activeElement as HTMLElement).blur(); - } - if (sline != null) { - let renderer = this.getRenderer(sline.lineid); - if (renderer != null) { - renderer.giveFocus(); - } - let termWrap = this.getTermWrap(sline.lineid); - if (termWrap != null) { - termWrap.giveFocus(); - } - } - } - - setFocusType(ftype: FocusTypeStrs): void { - mobx.action(() => { - this.focusType.set(ftype); - })(); - } - - setAnchor(anchorLine: number, anchorOffset: number): void { - let setVal = anchorLine == null || anchorLine == 0 ? "0" : sprintf("%d:%d", anchorLine, anchorOffset); - GlobalCommandRunner.screenSetAnchor(this.sessionId, this.screenId, setVal); - } - - getAnchor(): { anchorLine: number; anchorOffset: number } { - let anchor = this.anchor.get(); - if (anchor.anchorLine == null || anchor.anchorLine == 0) { - return { anchorLine: this.selectedLine.get(), anchorOffset: 0 }; - } - return anchor; - } - - getMaxLineNum(): number { - let win = this.getScreenLines(); - if (win == null) { - return null; - } - let lines = win.lines; - if (lines == null || lines.length == 0) { - return null; - } - return lines[lines.length - 1].linenum; - } - - getLineByNum(lineNum: number): LineType { - if (lineNum == null) { - return null; - } - let win = this.getScreenLines(); - if (win == null) { - return null; - } - let lines = win.lines; - if (lines == null || lines.length == 0) { - return null; - } - for (const line of lines) { - if (line.linenum == lineNum) { - return line; - } - } - return null; - } - - getLineById(lineId: string): LineType { - if (lineId == null) { - return null; - } - let win = this.getScreenLines(); - if (win == null) { - return null; - } - let lines = win.lines; - if (lines == null || lines.length == 0) { - return null; - } - for (const line of lines) { - if (line.lineid == lineId) { - return line; - } - } - return null; - } - - getPresentLineNum(lineNum: number): number { - let win = this.getScreenLines(); - if (win == null || !win.loaded.get()) { - return lineNum; - } - let lines = win.lines; - if (lines == null || lines.length == 0) { - return null; - } - if (lineNum == 0) { - return null; - } - for (const line of lines) { - if (line.linenum == lineNum) { - return lineNum; - } - if (line.linenum > lineNum) { - return line.linenum; - } - } - return lines[lines.length - 1].linenum; - } - - setSelectedLine(lineNum: number): void { - mobx.action(() => { - let pln = this.getPresentLineNum(lineNum); - if (pln != this.selectedLine.get()) { - this.selectedLine.set(pln); - } - })(); - } - - checkSelectedLine(): void { - let pln = this.getPresentLineNum(this.selectedLine.get()); - if (pln != this.selectedLine.get()) { - this.setSelectedLine(pln); - } - } - - updatePtyData(ptyMsg: PtyDataUpdateType) { - let lineId = ptyMsg.lineid; - let renderer = this.renderers[lineId]; - if (renderer != null) { - let data = base64ToArray(ptyMsg.ptydata64); - renderer.receiveData(ptyMsg.ptypos, data, "from-sw"); - } - let term = this.terminals[lineId]; - if (term != null) { - let data = base64ToArray(ptyMsg.ptydata64); - term.receiveData(ptyMsg.ptypos, data, "from-sw"); - } - } - - isActive(): boolean { - let activeScreen = GlobalModel.getActiveScreen(); - if (activeScreen == null) { - return false; - } - return this.sessionId == activeScreen.sessionId && this.screenId == activeScreen.screenId; - } - - screenSizeCallback(winSize: WindowSize): void { - if (winSize.height == 0 || winSize.width == 0) { - return; - } - if ( - this.lastScreenSize != null && - this.lastScreenSize.height == winSize.height && - this.lastScreenSize.width == winSize.width - ) { - return; - } - this.lastScreenSize = winSize; - let cols = windowWidthToCols(winSize.width, GlobalModel.termFontSize.get()); - let rows = windowHeightToRows(winSize.height, GlobalModel.termFontSize.get()); - this._termSizeCallback(rows, cols); - } - - getMaxContentSize(): WindowSize { - if (this.lastScreenSize == null) { - let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); - return { width, height }; - } - let winSize = this.lastScreenSize; - let minSize = MagicLayout.ScreenMinContentSize; - let maxSize = MagicLayout.ScreenMaxContentSize; - let width = boundInt(winSize.width - MagicLayout.ScreenMaxContentWidthBuffer, minSize, maxSize); - let height = boundInt(winSize.height - MagicLayout.ScreenMaxContentHeightBuffer, minSize, maxSize); - return { width, height }; - } - - getIdealContentSize(): WindowSize { - if (this.lastScreenSize == null) { - let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); - return { width, height }; - } - let winSize = this.lastScreenSize; - let width = boundInt(Math.ceil((winSize.width - 50) * 0.7), 100, 5000); - let height = boundInt(Math.ceil((winSize.height - 100) * 0.5), 100, 5000); - return { width, height }; - } - - _termSizeCallback(rows: number, cols: number): void { - if (cols == 0 || rows == 0) { - return; - } - if (rows == this.lastRows && cols == this.lastCols) { - return; - } - this.lastRows = rows; - this.lastCols = cols; - let exclude = []; - for (let lineid in this.terminals) { - let inSidebar = this.isLineIdInSidebar(lineid); - if (!inSidebar) { - this.terminals[lineid].resizeCols(cols); - } else { - exclude.push(lineid); - } - } - GlobalCommandRunner.resizeScreen(this.screenId, rows, cols, { exclude }); - } - - getTermWrap(lineId: string): TermWrap { - return this.terminals[lineId]; - } - - getRenderer(lineId: string): RendererModel { - return this.renderers[lineId]; - } - - registerRenderer(lineId: string, renderer: RendererModel) { - this.renderers[lineId] = renderer; - } - - setLineFocus(lineNum: number, focus: boolean): void { - mobx.action(() => this.termLineNumFocus.set(focus ? lineNum : 0))(); - if (focus && this.selectedLine.get() != lineNum) { - GlobalCommandRunner.screenSelectLine(String(lineNum), "cmd"); - } else if (focus && this.focusType.get() == "input") { - GlobalCommandRunner.screenSetFocus("cmd"); - } - } - - /** - * Set the status indicator for the screen. - * @param indicator The value of the status indicator. One of "none", "error", "success", "output". - */ - setStatusIndicator(indicator: StatusIndicatorLevel): void { - mobx.action(() => { - this.statusIndicator.set(indicator); - })(); - } - - /** - * Set the number of running commands for the screen. - * @param numRunning The number of running commands. - */ - setNumRunningCmds(numRunning: number): void { - mobx.action(() => { - this.numRunningCmds.set(numRunning); - })(); - } - - termCustomKeyHandlerInternal(e: any, termWrap: TermWrap): void { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if (checkKeyPressed(waveEvent, "ArrowUp")) { - termWrap.terminal.scrollLines(-1); - return; - } - if (checkKeyPressed(waveEvent, "ArrowDown")) { - termWrap.terminal.scrollLines(1); - return; - } - if (checkKeyPressed(waveEvent, "PageUp")) { - termWrap.terminal.scrollPages(-1); - return; - } - if (checkKeyPressed(waveEvent, "PageDown")) { - termWrap.terminal.scrollPages(1); - return; - } - } - - isTermCapturedKey(e: any): boolean { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if ( - checkKeyPressed(waveEvent, "ArrowUp") || - checkKeyPressed(waveEvent, "ArrowDown") || - checkKeyPressed(waveEvent, "PageUp") || - checkKeyPressed(waveEvent, "PageDown") - ) { - return true; - } - return false; - } - - termCustomKeyHandler(e: any, termWrap: TermWrap): boolean { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if (e.type == "keypress" && checkKeyPressed(waveEvent, "Ctrl:Shift:c")) { - e.stopPropagation(); - e.preventDefault(); - let sel = termWrap.terminal.getSelection(); - navigator.clipboard.writeText(sel); - return false; - } - if (e.type == "keypress" && checkKeyPressed(waveEvent, "Ctrl:Shift:v")) { - e.stopPropagation(); - e.preventDefault(); - let p = navigator.clipboard.readText(); - p.then((text) => { - termWrap.dataHandler?.(text, termWrap); - }); - return false; - } - if (termWrap.isRunning) { - return true; - } - let isCaptured = this.isTermCapturedKey(e); - if (!isCaptured) { - return true; - } - if (e.type != "keydown" || isModKeyPress(e)) { - return false; - } - e.stopPropagation(); - e.preventDefault(); - this.termCustomKeyHandlerInternal(e, termWrap); - return false; - } - - loadTerminalRenderer(elem: Element, line: LineType, cmd: Cmd, width: number) { - let lineId = cmd.lineId; - let termWrap = this.getTermWrap(lineId); - if (termWrap != null) { - console.log("term-wrap already exists for", this.screenId, lineId); - return; - } - let usedRows = GlobalModel.getContentHeight(getRendererContext(line)); - if (line.contentheight != null && line.contentheight != -1) { - usedRows = line.contentheight; - } - let termContext = { - sessionId: this.sessionId, - screenId: this.screenId, - lineId: line.lineid, - lineNum: line.linenum, - }; - termWrap = new TermWrap(elem, { - termContext: termContext, - usedRows: usedRows, - termOpts: cmd.getTermOpts(), - winSize: { height: 0, width: width }, - dataHandler: cmd.handleData.bind(cmd), - focusHandler: (focus: boolean) => this.setLineFocus(line.linenum, focus), - isRunning: cmd.isRunning(), - customKeyHandler: this.termCustomKeyHandler.bind(this), - fontSize: GlobalModel.termFontSize.get(), - ptyDataSource: getTermPtyData, - onUpdateContentHeight: (termContext: RendererContext, height: number) => { - GlobalModel.setContentHeight(termContext, height); - }, - }); - this.terminals[lineId] = termWrap; - if (this.focusType.get() == "cmd" && this.selectedLine.get() == line.linenum) { - termWrap.giveFocus(); - } - } - - unloadRenderer(lineId: string) { - let rmodel = this.renderers[lineId]; - if (rmodel != null) { - rmodel.dispose(); - delete this.renderers[lineId]; - } - let term = this.terminals[lineId]; - if (term != null) { - term.dispose(); - delete this.terminals[lineId]; - } - } - - getUsedRows(context: RendererContext, line: LineType, cmd: Cmd, width: number): number { - if (cmd == null) { - return 0; - } - let termOpts = cmd.getTermOpts(); - if (!termOpts.flexrows) { - return termOpts.rows; - } - let termWrap = this.getTermWrap(cmd.lineId); - if (termWrap == null) { - let usedRows = GlobalModel.getContentHeight(context); - if (usedRows != null) { - return usedRows; - } - if (line.contentheight != null && line.contentheight != -1) { - return line.contentheight; - } - return cmd.isRunning() ? 1 : 0; - } - return termWrap.getUsedRows(); - } - - getIsFocused(lineNum: number): boolean { - return this.termLineNumFocus.get() == lineNum; - } - - getSelectedLine(): number { - return this.selectedLine.get(); - } - - getScreenLines(): ScreenLines { - return GlobalModel.getScreenLinesById(this.screenId); - } - - getFocusType(): FocusTypeStrs { - return this.focusType.get(); - } - - giveFocus(): void { - if (!this.isActive()) { - return; - } - let ftype = this.focusType.get(); - if (ftype == "input") { - GlobalModel.inputModel.giveFocus(); - } else { - let sline: LineType = null; - if (this.selectedLine.get() != 0) { - sline = this.getLineByNum(this.selectedLine.get()); - } - if (sline != null) { - let renderer = this.getRenderer(sline.lineid); - if (renderer != null) { - renderer.giveFocus(); - } - let termWrap = this.getTermWrap(sline.lineid); - if (termWrap != null) { - termWrap.giveFocus(); - } - } - } - } -} - -class ScreenLines { - screenId: string; - loaded: OV = mobx.observable.box(false, { name: "slines-loaded" }); - loadError: OV = mobx.observable.box(null); - lines: OArr = mobx.observable.array([], { - name: "slines-lines", - deep: false, - }); - cmds: Record = {}; // lineid => Cmd - - constructor(screenId: string) { - this.screenId = screenId; - } - - getNonArchivedLines(): LineType[] { - let rtn: LineType[] = []; - for (const line of this.lines) { - if (line.archived) { - continue; - } - rtn.push(line); - } - return rtn; - } - - updateData(slines: ScreenLinesType, load: boolean) { - mobx.action(() => { - if (load) { - this.loaded.set(true); - } - genMergeSimpleData( - this.lines, - slines.lines, - (l: LineType) => String(l.lineid), - (l: LineType) => sprintf("%013d:%s", l.ts, l.lineid) - ); - let cmds = slines.cmds || []; - for (const cmd of cmds) { - this.cmds[cmd.lineid] = new Cmd(cmd); - } - })(); - } - - setLoadError(errStr: string) { - mobx.action(() => { - this.loaded.set(true); - this.loadError.set(errStr); - })(); - } - - dispose() {} - - getCmd(lineId: string): Cmd { - return this.cmds[lineId]; - } - - /** - * Get all running cmds in the screen. - * @param returnFirst If true, return the first running cmd found. - * @returns An array of running cmds, or the first running cmd if returnFirst is true. - */ - getRunningCmdLines(returnFirst?: boolean): LineType[] { - let rtn: LineType[] = []; - for (const line of this.lines) { - const cmd = this.getCmd(line.lineid); - if (cmd == null) { - continue; - } - const status = cmd.getStatus(); - if (cmdStatusIsRunning(status)) { - if (returnFirst) { - return [line]; - } - rtn.push(line); - } - } - return rtn; - } - - /** - * Check if there are any running cmds in the screen. - * @returns True if there are any running cmds. - */ - hasRunningCmdLines(): boolean { - return this.getRunningCmdLines(true).length > 0; - } - - updateCmd(cmd: CmdDataType): void { - if (cmd.remove) { - throw new Error("cannot remove cmd with updateCmd call [" + cmd.lineid + "]"); - } - let origCmd = this.cmds[cmd.lineid]; - if (origCmd != null) { - origCmd.setCmd(cmd); - } - } - - mergeCmd(cmd: CmdDataType): void { - if (cmd.remove) { - delete this.cmds[cmd.lineid]; - return; - } - let origCmd = this.cmds[cmd.lineid]; - if (origCmd == null) { - this.cmds[cmd.lineid] = new Cmd(cmd); - return; - } - origCmd.setCmd(cmd); - } - - addLineCmd(line: LineType, cmd: CmdDataType, interactive: boolean) { - if (!this.loaded.get()) { - return; - } - mobx.action(() => { - if (cmd != null) { - this.mergeCmd(cmd); - } - if (line != null) { - let lines = this.lines; - if (line.remove) { - for (let i = 0; i < lines.length; i++) { - if (lines[i].lineid == line.lineid) { - this.lines.splice(i, 1); - break; - } - } - return; - } - let lineIdx = 0; - for (lineIdx = 0; lineIdx < lines.length; lineIdx++) { - let lineId = lines[lineIdx].lineid; - let curTs = lines[lineIdx].ts; - if (lineId == line.lineid) { - this.lines[lineIdx] = line; - return; - } - if (curTs > line.ts || (curTs == line.ts && lineId > line.lineid)) { - break; - } - } - if (lineIdx == lines.length) { - this.lines.push(line); - return; - } - this.lines.splice(lineIdx, 0, line); - } - })(); - } -} - -class Session { - sessionId: string; - name: OV; - activeScreenId: OV; - sessionIdx: OV; - notifyNum: OV = mobx.observable.box(0); - remoteInstances: OArr; - archived: OV; - - constructor(sdata: SessionDataType) { - this.sessionId = sdata.sessionid; - this.name = mobx.observable.box(sdata.name); - this.sessionIdx = mobx.observable.box(sdata.sessionidx); - this.archived = mobx.observable.box(!!sdata.archived); - this.activeScreenId = mobx.observable.box(ces(sdata.activescreenid)); - let remotes = sdata.remotes || []; - this.remoteInstances = mobx.observable.array(remotes); - } - - dispose(): void {} - - // session updates only contain screens (no windows) - mergeData(sdata: SessionDataType) { - if (sdata.sessionid != this.sessionId) { - throw new Error( - sprintf( - "cannot merge session data, sessionids don't match sid=%s, data-sid=%s", - this.sessionId, - sdata.sessionid - ) - ); - } - mobx.action(() => { - if (!isBlank(sdata.name)) { - this.name.set(sdata.name); - } - if (sdata.sessionidx > 0) { - this.sessionIdx.set(sdata.sessionidx); - } - if (sdata.notifynum >= 0) { - this.notifyNum.set(sdata.notifynum); - } - this.archived.set(!!sdata.archived); - if (!isBlank(sdata.activescreenid)) { - let screen = this.getScreenById(sdata.activescreenid); - if (screen == null) { - console.log( - sprintf("got session update, activescreenid=%s, screen not found", sdata.activescreenid) - ); - } else { - this.activeScreenId.set(sdata.activescreenid); - } - } - genMergeSimpleData(this.remoteInstances, sdata.remotes, (r) => r.riid, null); - })(); - } - - getActiveScreen(): Screen { - return this.getScreenById(this.activeScreenId.get()); - } - - setActiveScreenId(screenId: string) { - this.activeScreenId.set(screenId); - } - - getScreenById(screenId: string): Screen { - if (screenId == null) { - return null; - } - return GlobalModel.getScreenById(this.sessionId, screenId); - } - - getRemoteInstance(screenId: string, rptr: RemotePtrType): RemoteInstanceType { - if (rptr.name.startsWith("*")) { - screenId = ""; - } - for (const rdata of this.remoteInstances) { - if ( - rdata.screenid == screenId && - rdata.remoteid == rptr.remoteid && - rdata.remoteownerid == rptr.ownerid && - rdata.name == rptr.name - ) { - return rdata; - } - } - let remote = GlobalModel.getRemote(rptr.remoteid); - if (remote != null) { - return { - riid: "", - sessionid: this.sessionId, - screenid: screenId, - remoteownerid: rptr.ownerid, - remoteid: rptr.remoteid, - name: rptr.name, - festate: remote.defaultfestate, - shelltype: remote.defaultshelltype, - }; - } - return null; - } -} - -function getDefaultHistoryQueryOpts(): HistoryQueryOpts { - return { - queryType: "screen", - limitRemote: true, - limitRemoteInstance: true, - limitUser: true, - queryStr: "", - maxItems: 10000, - includeMeta: true, - fromTs: 0, - }; -} - -class InputModel { - historyShow: OV = mobx.observable.box(false); - infoShow: OV = mobx.observable.box(false); - aIChatShow: OV = mobx.observable.box(false); - cmdInputHeight: OV = mobx.observable.box(0); - aiChatTextAreaRef: React.RefObject; - aiChatWindowRef: React.RefObject; - codeSelectBlockRefArray: Array>; - codeSelectSelectedIndex: OV = mobx.observable.box(-1); - - AICmdInfoChatItems: mobx.IObservableArray = mobx.observable.array([], { - name: "aicmdinfo-chat", - }); - readonly codeSelectTop: number = -2; - readonly codeSelectBottom: number = -1; - - historyType: mobx.IObservableValue = mobx.observable.box("screen"); - historyLoading: mobx.IObservableValue = mobx.observable.box(false); - historyAfterLoadIndex: number = 0; - historyItems: mobx.IObservableValue = mobx.observable.box(null, { - name: "history-items", - deep: false, - }); // sorted in reverse (most recent is index 0) - filteredHistoryItems: mobx.IComputedValue = null; - historyIndex: mobx.IObservableValue = mobx.observable.box(0, { - name: "history-index", - }); // 1-indexed (because 0 is current) - modHistory: mobx.IObservableArray = mobx.observable.array([""], { - name: "mod-history", - }); - historyQueryOpts: OV = mobx.observable.box(getDefaultHistoryQueryOpts()); - - infoMsg: OV = mobx.observable.box(null); - infoTimeoutId: any = null; - inputMode: OV = mobx.observable.box(null); - inputExpanded: OV = mobx.observable.box(false, { - name: "inputExpanded", - }); - - // cursor - forceCursorPos: OV = mobx.observable.box(null); - - // focus - inputFocused: OV = mobx.observable.box(false); - lineFocused: OV = mobx.observable.box(false); - physicalInputFocused: OV = mobx.observable.box(false); - forceInputFocus: boolean = false; - - constructor() { - this.filteredHistoryItems = mobx.computed(() => { - return this._getFilteredHistoryItems(); - }); - mobx.action(() => { - this.codeSelectSelectedIndex.set(-1); - this.codeSelectBlockRefArray = []; - })(); - } - - setInputMode(inputMode: null | "comment" | "global"): void { - mobx.action(() => { - this.inputMode.set(inputMode); - })(); - } - - onInputFocus(isFocused: boolean): void { - mobx.action(() => { - if (isFocused) { - this.inputFocused.set(true); - this.lineFocused.set(false); - } else if (this.inputFocused.get()) { - this.inputFocused.set(false); - } - })(); - } - - onLineFocus(isFocused: boolean): void { - mobx.action(() => { - if (isFocused) { - this.inputFocused.set(false); - this.lineFocused.set(true); - } else if (this.lineFocused.get()) { - this.lineFocused.set(false); - } - })(); - } - - _focusCmdInput(): void { - let elem = document.getElementById("main-cmd-input"); - if (elem != null) { - elem.focus(); - } - } - - _focusHistoryInput(): void { - let elem: HTMLElement = document.querySelector(".cmd-input input.history-input"); - if (elem != null) { - elem.focus(); - } - } - - giveFocus(): void { - if (this.historyShow.get()) { - this._focusHistoryInput(); - } else { - this._focusCmdInput(); - } - } - - setPhysicalInputFocused(isFocused: boolean): void { - mobx.action(() => { - this.physicalInputFocused.set(isFocused); - })(); - if (isFocused) { - let screen = GlobalModel.getActiveScreen(); - if (screen != null) { - if (screen.focusType.get() != "input") { - GlobalCommandRunner.screenSetFocus("input"); - } - } - } - } - - hasFocus(): boolean { - let mainInputElem = document.getElementById("main-cmd-input"); - if (document.activeElement == mainInputElem) { - return true; - } - let historyInputElem = document.querySelector(".cmd-input input.history-input"); - if (document.activeElement == historyInputElem) { - return true; - } - return false; - } - - setHistoryType(htype: HistoryTypeStrs): void { - if (this.historyQueryOpts.get().queryType == htype) { - return; - } - this.loadHistory(true, -1, htype); - } - - findBestNewIndex(oldItem: HistoryItem): number { - if (oldItem == null) { - return 0; - } - let newItems = this.getFilteredHistoryItems(); - if (newItems.length == 0) { - return 0; - } - let bestIdx = 0; - for (let i = 0; i < newItems.length; i++) { - // still start at i=0 to catch the historynum equality case - let item = newItems[i]; - if (item.historynum == oldItem.historynum) { - bestIdx = i; - break; - } - let bestTsDiff = Math.abs(item.ts - newItems[bestIdx].ts); - let curTsDiff = Math.abs(item.ts - oldItem.ts); - if (curTsDiff < bestTsDiff) { - bestIdx = i; - } - } - return bestIdx + 1; - } - - setHistoryQueryOpts(opts: HistoryQueryOpts): void { - mobx.action(() => { - let oldItem = this.getHistorySelectedItem(); - this.historyQueryOpts.set(opts); - let bestIndex = this.findBestNewIndex(oldItem); - setTimeout(() => this.setHistoryIndex(bestIndex, true), 10); - return; - })(); - } - - setOpenAICmdInfoChat(chat: OpenAICmdInfoChatMessageType[]): void { - this.AICmdInfoChatItems.replace(chat); - this.codeSelectBlockRefArray = []; - } - - setHistoryShow(show: boolean): void { - if (this.historyShow.get() == show) { - return; - } - mobx.action(() => { - this.historyShow.set(show); - if (this.hasFocus()) { - this.giveFocus(); - } - })(); - } - - isHistoryLoaded(): boolean { - if (this.historyLoading.get()) { - return false; - } - let hitems = this.historyItems.get(); - return hitems != null; - } - - loadHistory(show: boolean, afterLoadIndex: number, htype: HistoryTypeStrs) { - if (this.historyLoading.get()) { - return; - } - if (this.isHistoryLoaded()) { - if (this.historyQueryOpts.get().queryType == htype) { - return; - } - } - this.historyAfterLoadIndex = afterLoadIndex; - mobx.action(() => { - this.historyLoading.set(true); - })(); - GlobalCommandRunner.loadHistory(show, htype); - } - - openHistory(): void { - if (this.historyLoading.get()) { - return; - } - if (!this.isHistoryLoaded()) { - this.loadHistory(true, 0, "screen"); - return; - } - if (!this.historyShow.get()) { - mobx.action(() => { - this.setHistoryShow(true); - this.infoShow.set(false); - this.dropModHistory(true); - this.giveFocus(); - })(); - } - } - - updateCmdLine(cmdLine: T.StrWithPos): void { - mobx.action(() => { - this.setCurLine(cmdLine.str); - if (cmdLine.pos != appconst.NoStrPos) { - this.forceCursorPos.set(cmdLine.pos); - } - })(); - } - - getHistorySelectedItem(): HistoryItem { - let hidx = this.historyIndex.get(); - if (hidx == 0) { - return null; - } - let hitems = this.getFilteredHistoryItems(); - if (hidx > hitems.length) { - return null; - } - return hitems[hidx - 1]; - } - - getFirstHistoryItem(): HistoryItem { - let hitems = this.getFilteredHistoryItems(); - if (hitems.length == 0) { - return null; - } - return hitems[0]; - } - - setHistorySelectionNum(hnum: string): void { - let hitems = this.getFilteredHistoryItems(); - for (let i = 0; i < hitems.length; i++) { - if (hitems[i].historynum == hnum) { - this.setHistoryIndex(i + 1); - return; - } - } - } - - setHistoryInfo(hinfo: HistoryInfoType): void { - mobx.action(() => { - let oldItem = this.getHistorySelectedItem(); - let hitems: HistoryItem[] = hinfo.items ?? []; - this.historyItems.set(hitems); - this.historyLoading.set(false); - this.historyQueryOpts.get().queryType = hinfo.historytype; - if (hinfo.historytype == "session" || hinfo.historytype == "global") { - this.historyQueryOpts.get().limitRemote = false; - this.historyQueryOpts.get().limitRemoteInstance = false; - } - if (this.historyAfterLoadIndex == -1) { - let bestIndex = this.findBestNewIndex(oldItem); - setTimeout(() => this.setHistoryIndex(bestIndex, true), 100); - } else if (this.historyAfterLoadIndex) { - if (hitems.length >= this.historyAfterLoadIndex) { - this.setHistoryIndex(this.historyAfterLoadIndex); - } - } - this.historyAfterLoadIndex = 0; - if (hinfo.show) { - this.openHistory(); - } - })(); - } - - getFilteredHistoryItems(): HistoryItem[] { - return this.filteredHistoryItems.get(); - } - - _getFilteredHistoryItems(): HistoryItem[] { - let hitems: HistoryItem[] = this.historyItems.get() ?? []; - let rtn: HistoryItem[] = []; - let opts = mobx.toJS(this.historyQueryOpts.get()); - let ctx = GlobalModel.getUIContext(); - let curRemote: RemotePtrType = ctx.remote; - if (curRemote == null) { - curRemote = { ownerid: "", name: "", remoteid: "" }; - } - curRemote = mobx.toJS(curRemote); - for (const hitem of hitems) { - if (hitem.ismetacmd) { - if (!opts.includeMeta) { - continue; - } - } else if (opts.limitRemoteInstance) { - if (hitem.remote == null || isBlank(hitem.remote.remoteid)) { - continue; - } - if ( - (curRemote.ownerid ?? "") != (hitem.remote.ownerid ?? "") || - (curRemote.remoteid ?? "") != (hitem.remote.remoteid ?? "") || - (curRemote.name ?? "") != (hitem.remote.name ?? "") - ) { - continue; - } - } else if (opts.limitRemote) { - if (hitem.remote == null || isBlank(hitem.remote.remoteid)) { - continue; - } - if ( - (curRemote.ownerid ?? "") != (hitem.remote.ownerid ?? "") || - (curRemote.remoteid ?? "") != (hitem.remote.remoteid ?? "") - ) { - continue; - } - } - if (!isBlank(opts.queryStr)) { - if (isBlank(hitem.cmdstr)) { - continue; - } - let idx = hitem.cmdstr.indexOf(opts.queryStr); - if (idx == -1) { - continue; - } - } - - rtn.push(hitem); - } - return rtn; - } - - scrollHistoryItemIntoView(hnum: string): void { - let elem: HTMLElement = document.querySelector(".cmd-history .hnum-" + hnum); - if (elem == null) { - return; - } - let historyDiv = elem.closest(".cmd-history"); - if (historyDiv == null) { - return; - } - let buffer = 15; - let titleHeight = 24; - let titleDiv: HTMLElement = document.querySelector(".cmd-history .history-title"); - if (titleDiv != null) { - titleHeight = titleDiv.offsetHeight + 2; - } - let elemOffset = elem.offsetTop; - let elemHeight = elem.clientHeight; - let topPos = historyDiv.scrollTop; - let endPos = topPos + historyDiv.clientHeight; - if (elemOffset + elemHeight + buffer > endPos) { - if (elemHeight + buffer > historyDiv.clientHeight - titleHeight) { - historyDiv.scrollTop = elemOffset - titleHeight; - return; - } - historyDiv.scrollTop = elemOffset - historyDiv.clientHeight + elemHeight + buffer; - return; - } - if (elemOffset < topPos + titleHeight) { - if (elemHeight + buffer > historyDiv.clientHeight - titleHeight) { - historyDiv.scrollTop = elemOffset - titleHeight; - return; - } - historyDiv.scrollTop = elemOffset - titleHeight - buffer; - } - } - - grabSelectedHistoryItem(): void { - let hitem = this.getHistorySelectedItem(); - if (hitem == null) { - this.resetHistory(); - return; - } - mobx.action(() => { - this.resetInput(); - this.setCurLine(hitem.cmdstr); - })(); - } - - setHistoryIndex(hidx: number, force?: boolean): void { - if (hidx < 0) { - return; - } - if (!force && this.historyIndex.get() == hidx) { - return; - } - mobx.action(() => { - this.historyIndex.set(hidx); - if (this.historyShow.get()) { - let hitem = this.getHistorySelectedItem(); - if (hitem == null) { - hitem = this.getFirstHistoryItem(); - } - if (hitem != null) { - this.scrollHistoryItemIntoView(hitem.historynum); - } - } - })(); - } - - moveHistorySelection(amt: number): void { - if (amt == 0) { - return; - } - if (!this.isHistoryLoaded()) { - return; - } - let hitems = this.getFilteredHistoryItems(); - let idx = this.historyIndex.get(); - idx += amt; - if (idx < 0) { - idx = 0; - } - if (idx > hitems.length) { - idx = hitems.length; - } - this.setHistoryIndex(idx); - } - - flashInfoMsg(info: InfoType, timeoutMs: number): void { - this._clearInfoTimeout(); - mobx.action(() => { - this.infoMsg.set(info); - if (info == null) { - this.infoShow.set(false); - } else { - this.infoShow.set(true); - this.setHistoryShow(false); - } - })(); - if (info != null && timeoutMs) { - this.infoTimeoutId = setTimeout(() => { - if (this.historyShow.get()) { - return; - } - this.clearInfoMsg(false); - }, timeoutMs); - } - } - - setCmdInfoChatRefs( - textAreaRef: React.RefObject, - chatWindowRef: React.RefObject - ) { - this.aiChatTextAreaRef = textAreaRef; - this.aiChatWindowRef = chatWindowRef; - } - - setAIChatFocus() { - if (this.aiChatTextAreaRef?.current != null) { - this.aiChatTextAreaRef.current.focus(); - } - } - - grabCodeSelectSelection() { - if ( - this.codeSelectSelectedIndex.get() >= 0 && - this.codeSelectSelectedIndex.get() < this.codeSelectBlockRefArray.length - ) { - let curBlockRef = this.codeSelectBlockRefArray[this.codeSelectSelectedIndex.get()]; - let codeText = curBlockRef.current.innerText; - codeText = codeText.replace(/\n$/, ""); // remove trailing newline - this.setCurLine(codeText); - this.giveFocus(); - } - } - - addCodeBlockToCodeSelect(blockRef: React.RefObject): number { - let rtn = -1; - rtn = this.codeSelectBlockRefArray.length; - this.codeSelectBlockRefArray.push(blockRef); - return rtn; - } - - setCodeSelectSelectedCodeBlock(blockIndex: number) { - mobx.action(() => { - if (blockIndex >= 0 && blockIndex < this.codeSelectBlockRefArray.length) { - this.codeSelectSelectedIndex.set(blockIndex); - let currentRef = this.codeSelectBlockRefArray[blockIndex].current; - if (currentRef != null) { - if (this.aiChatWindowRef?.current != null) { - let chatWindowTop = this.aiChatWindowRef.current.scrollTop; - let chatWindowBottom = chatWindowTop + this.aiChatWindowRef.current.clientHeight - 100; - let elemTop = currentRef.offsetTop; - let elemBottom = elemTop - currentRef.offsetHeight; - let elementIsInView = elemBottom < chatWindowBottom && elemTop > chatWindowTop; - if (!elementIsInView) { - this.aiChatWindowRef.current.scrollTop = - elemBottom - this.aiChatWindowRef.current.clientHeight / 3; - } - } - } - this.codeSelectBlockRefArray = []; - this.setAIChatFocus(); - } - })(); - } - - codeSelectSelectNextNewestCodeBlock() { - // oldest code block = index 0 in array - // this decrements codeSelectSelected index - mobx.action(() => { - if (this.codeSelectSelectedIndex.get() == this.codeSelectTop) { - this.codeSelectSelectedIndex.set(this.codeSelectBottom); - } else if (this.codeSelectSelectedIndex.get() == this.codeSelectBottom) { - return; - } - let incBlockIndex = this.codeSelectSelectedIndex.get() + 1; - if (this.codeSelectSelectedIndex.get() == this.codeSelectBlockRefArray.length - 1) { - this.codeSelectDeselectAll(); - if (this.aiChatWindowRef?.current != null) { - this.aiChatWindowRef.current.scrollTop = this.aiChatWindowRef.current.scrollHeight; - } - } - if (incBlockIndex >= 0 && incBlockIndex < this.codeSelectBlockRefArray.length) { - this.setCodeSelectSelectedCodeBlock(incBlockIndex); - } - })(); - } - - codeSelectSelectNextOldestCodeBlock() { - mobx.action(() => { - if (this.codeSelectSelectedIndex.get() == this.codeSelectBottom) { - if (this.codeSelectBlockRefArray.length > 0) { - this.codeSelectSelectedIndex.set(this.codeSelectBlockRefArray.length); - } else { - return; - } - } else if (this.codeSelectSelectedIndex.get() == this.codeSelectTop) { - return; - } - let decBlockIndex = this.codeSelectSelectedIndex.get() - 1; - if (decBlockIndex < 0) { - this.codeSelectDeselectAll(this.codeSelectTop); - if (this.aiChatWindowRef?.current != null) { - this.aiChatWindowRef.current.scrollTop = 0; - } - } - if (decBlockIndex >= 0 && decBlockIndex < this.codeSelectBlockRefArray.length) { - this.setCodeSelectSelectedCodeBlock(decBlockIndex); - } - })(); - } - - getCodeSelectSelectedIndex() { - return this.codeSelectSelectedIndex.get(); - } - - getCodeSelectRefArrayLength() { - return this.codeSelectBlockRefArray.length; - } - - codeBlockIsSelected(blockIndex: number): boolean { - return blockIndex == this.codeSelectSelectedIndex.get(); - } - - codeSelectDeselectAll(direction: number = this.codeSelectBottom) { - mobx.action(() => { - this.codeSelectSelectedIndex.set(direction); - this.codeSelectBlockRefArray = []; - })(); - } - - openAIAssistantChat(): void { - this.aIChatShow.set(true); - this.setAIChatFocus(); - } - - closeAIAssistantChat(): void { - this.aIChatShow.set(false); - this.giveFocus(); - } - - clearAIAssistantChat(): void { - let prtn = GlobalModel.submitChatInfoCommand("", "", true); - prtn.then((rtn) => { - if (!rtn.success) { - console.log("submit chat command error: " + rtn.error); - } - }).catch((error) => { - console.log("submit chat command error: ", error); - }); - } - - hasScrollingInfoMsg(): boolean { - if (!this.infoShow.get()) { - return false; - } - let info = this.infoMsg.get(); - if (info == null) { - return false; - } - let div = document.querySelector(".cmd-input-info"); - if (div == null) { - return false; - } - return div.scrollHeight > div.clientHeight; - } - - _clearInfoTimeout(): void { - if (this.infoTimeoutId != null) { - clearTimeout(this.infoTimeoutId); - this.infoTimeoutId = null; - } - } - - clearInfoMsg(setNull: boolean): void { - this._clearInfoTimeout(); - mobx.action(() => { - this.setHistoryShow(false); - this.infoShow.set(false); - if (setNull) { - this.infoMsg.set(null); - } - })(); - } - - toggleInfoMsg(): void { - this._clearInfoTimeout(); - mobx.action(() => { - if (this.historyShow.get()) { - this.setHistoryShow(false); - return; - } - let isShowing = this.infoShow.get(); - if (isShowing) { - this.infoShow.set(false); - } else { - if (this.infoMsg.get() != null) { - this.infoShow.set(true); - } - } - })(); - } - - @boundMethod - uiSubmitCommand(): void { - mobx.action(() => { - let commandStr = this.getCurLine(); - if (commandStr.trim() == "") { - return; - } - this.resetInput(); - GlobalModel.submitRawCommand(commandStr, true, true); - })(); - } - - isEmpty(): boolean { - return this.getCurLine().trim() == ""; - } - - resetInputMode(): void { - mobx.action(() => { - this.setInputMode(null); - this.setCurLine(""); - })(); - } - - setCurLine(val: string): void { - let hidx = this.historyIndex.get(); - mobx.action(() => { - // if (val == "\" ") { - // this.setInputMode("comment"); - // val = ""; - // } - // if (val == "//") { - // this.setInputMode("global"); - // val = ""; - // } - if (this.modHistory.length <= hidx) { - this.modHistory.length = hidx + 1; - } - this.modHistory[hidx] = val; - })(); - } - - resetInput(): void { - mobx.action(() => { - this.setHistoryShow(false); - this.closeAIAssistantChat(); - this.infoShow.set(false); - this.inputMode.set(null); - this.resetHistory(); - this.dropModHistory(false); - this.infoMsg.set(null); - this.inputExpanded.set(false); - this._clearInfoTimeout(); - })(); - } - - @boundMethod - toggleExpandInput(): void { - mobx.action(() => { - this.inputExpanded.set(!this.inputExpanded.get()); - this.forceInputFocus = true; - })(); - } - - getCurLine(): string { - let hidx = this.historyIndex.get(); - if (hidx < this.modHistory.length && this.modHistory[hidx] != null) { - return this.modHistory[hidx]; - } - let hitems = this.getFilteredHistoryItems(); - if (hidx == 0 || hitems == null || hidx > hitems.length) { - return ""; - } - let hitem = hitems[hidx - 1]; - if (hitem == null) { - return ""; - } - return hitem.cmdstr; - } - - dropModHistory(keepLine0: boolean): void { - mobx.action(() => { - if (keepLine0) { - if (this.modHistory.length > 1) { - this.modHistory.splice(1, this.modHistory.length - 1); - } - } else { - this.modHistory.replace([""]); - } - })(); - } - - resetHistory(): void { - mobx.action(() => { - this.setHistoryShow(false); - this.historyLoading.set(false); - this.historyType.set("screen"); - this.historyItems.set(null); - this.historyIndex.set(0); - this.historyQueryOpts.set(getDefaultHistoryQueryOpts()); - this.historyAfterLoadIndex = 0; - this.dropModHistory(true); - })(); - } -} - -type LineFocusType = { - cmdInputFocus: boolean; - lineid?: string; - linenum?: number; - screenid?: string; -}; - -type CmdFinder = { - getCmdById(cmdId: string): Cmd; -}; - -class ForwardLineContainer { - winSize: T.WindowSize; - screen: Screen; - containerType: T.LineContainerStrs; - lineId: string; - - constructor(screen: Screen, winSize: T.WindowSize, containerType: T.LineContainerStrs, lineId: string) { - this.screen = screen; - this.winSize = winSize; - this.containerType = containerType; - this.lineId = lineId; - } - - screenSizeCallback(winSize: WindowSize): void { - this.winSize = winSize; - let termWrap = this.getTermWrap(this.lineId); - if (termWrap != null) { - let fontSize = GlobalModel.termFontSize.get(); - let cols = windowWidthToCols(winSize.width, fontSize); - let rows = windowHeightToRows(winSize.height, fontSize); - termWrap.resizeCols(cols); - GlobalCommandRunner.resizeScreen(this.screen.screenId, rows, cols, { include: [this.lineId] }); - } - } - - getContainerType(): T.LineContainerStrs { - return this.containerType; - } - - getCmd(line: LineType): Cmd { - return this.screen.getCmd(line); - } - - isSidebarOpen(): boolean { - return false; - } - - isLineIdInSidebar(lineId: string): boolean { - return false; - } - - setLineFocus(lineNum: number, focus: boolean): void { - this.screen.setLineFocus(lineNum, focus); - } - - setContentHeight(context: RendererContext, height: number): void { - return; - } - - getMaxContentSize(): WindowSize { - let rtn = { width: this.winSize.width, height: this.winSize.height }; - rtn.width = rtn.width - MagicLayout.ScreenMaxContentWidthBuffer; - return rtn; - } - - getIdealContentSize(): WindowSize { - return this.winSize; - } - - loadTerminalRenderer(elem: Element, line: LineType, cmd: Cmd, width: number): void { - this.screen.loadTerminalRenderer(elem, line, cmd, width); - } - - registerRenderer(lineId: string, renderer: RendererModel): void { - this.screen.registerRenderer(lineId, renderer); - } - - unloadRenderer(lineId: string): void { - this.screen.unloadRenderer(lineId); - } - - getContentHeight(context: RendererContext): number { - return this.screen.getContentHeight(context); - } - - getUsedRows(context: RendererContext, line: LineType, cmd: Cmd, width: number): number { - return this.screen.getUsedRows(context, line, cmd, width); - } - - getIsFocused(lineNum: number): boolean { - return this.screen.getIsFocused(lineNum); - } - - getRenderer(lineId: string): RendererModel { - return this.screen.getRenderer(lineId); - } - - getTermWrap(lineId: string): TermWrap { - return this.screen.getTermWrap(lineId); - } - - getFocusType(): FocusTypeStrs { - return this.screen.getFocusType(); - } - - getSelectedLine(): number { - return this.screen.getSelectedLine(); - } -} - -class SpecialLineContainer { - wsize: T.WindowSize; - allowInput: boolean; - terminal: TermWrap; - renderer: RendererModel; - cmd: Cmd; - cmdFinder: CmdFinder; - containerType: T.LineContainerStrs; - - constructor(cmdFinder: CmdFinder, wsize: T.WindowSize, allowInput: boolean, containerType: T.LineContainerStrs) { - this.cmdFinder = cmdFinder; - this.wsize = wsize; - this.allowInput = allowInput; - } - - getCmd(line: LineType): Cmd { - if (this.cmd == null) { - this.cmd = this.cmdFinder.getCmdById(line.lineid); - } - return this.cmd; - } - - getContainerType(): T.LineContainerStrs { - return this.containerType; - } - - isSidebarOpen(): boolean { - return false; - } - - isLineIdInSidebar(lineId: string): boolean { - return false; - } - - setLineFocus(lineNum: number, focus: boolean): void { - return; - } - - setContentHeight(context: RendererContext, height: number): void { - return; - } - - getMaxContentSize(): WindowSize { - return this.wsize; - } - - getIdealContentSize(): WindowSize { - return this.wsize; - } - - loadTerminalRenderer(elem: Element, line: LineType, cmd: Cmd, width: number): void { - this.unloadRenderer(null); - let lineId = cmd.lineId; - let termWrap = this.getTermWrap(lineId); - if (termWrap != null) { - console.log("term-wrap already exists for", line.screenid, lineId); - return; - } - let usedRows = GlobalModel.getContentHeight(getRendererContext(line)); - if (line.contentheight != null && line.contentheight != -1) { - usedRows = line.contentheight; - } - let termContext = { - screenId: line.screenid, - lineId: line.lineid, - lineNum: line.linenum, - }; - termWrap = new TermWrap(elem, { - termContext: termContext, - usedRows: usedRows, - termOpts: cmd.getTermOpts(), - winSize: { height: 0, width: width }, - dataHandler: null, - focusHandler: null, - isRunning: cmd.isRunning(), - customKeyHandler: null, - fontSize: GlobalModel.termFontSize.get(), - ptyDataSource: getTermPtyData, - onUpdateContentHeight: null, - }); - this.terminal = termWrap; - } - - registerRenderer(lineId: string, renderer: RendererModel): void { - this.renderer = renderer; - } - - unloadRenderer(lineId: string): void { - if (this.renderer != null) { - this.renderer.dispose(); - this.renderer = null; - } - if (this.terminal != null) { - this.terminal.dispose(); - this.terminal = null; - } - } - - getContentHeight(context: RendererContext): number { - return GlobalModel.getContentHeight(context); - } - - getUsedRows(context: RendererContext, line: LineType, cmd: Cmd, width: number): number { - if (cmd == null) { - return 0; - } - let termOpts = cmd.getTermOpts(); - if (!termOpts.flexrows) { - return termOpts.rows; - } - let termWrap = this.getTermWrap(cmd.lineId); - if (termWrap == null) { - let cols = windowWidthToCols(width, GlobalModel.termFontSize.get()); - let usedRows = GlobalModel.getContentHeight(context); - if (usedRows != null) { - return usedRows; - } - if (line.contentheight != null && line.contentheight != -1) { - return line.contentheight; - } - return cmd.isRunning() ? 1 : 0; - } - return termWrap.getUsedRows(); - } - - getIsFocused(lineNum: number): boolean { - return false; - } - - getRenderer(lineId: string): RendererModel { - return this.renderer; - } - - getTermWrap(lineId: string): TermWrap { - return this.terminal; - } - - getFocusType(): FocusTypeStrs { - return "input"; - } - - getSelectedLine(): number { - return null; - } -} - -const HistoryPageSize = 50; - -class HistoryViewModel { - items: OArr = mobx.observable.array([], { - name: "HistoryItems", - }); - hasMore: OV = mobx.observable.box(false, { - name: "historyview-hasmore", - }); - offset: OV = mobx.observable.box(0, { name: "historyview-offset" }); - searchText: OV = mobx.observable.box("", { - name: "historyview-searchtext", - }); - activeSearchText: string = null; - selectedItems: OMap = mobx.observable.map({}, { name: "historyview-selectedItems" }); - deleteActive: OV = mobx.observable.box(false, { - name: "historyview-deleteActive", - }); - activeItem: OV = mobx.observable.box(null, { - name: "historyview-activeItem", - }); - searchSessionId: OV = mobx.observable.box(null, { - name: "historyview-searchSessionId", - }); - searchRemoteId: OV = mobx.observable.box(null, { - name: "historyview-searchRemoteId", - }); - searchShowMeta: OV = mobx.observable.box(true, { - name: "historyview-searchShowMeta", - }); - searchFromDate: OV = mobx.observable.box(null, { - name: "historyview-searchfromts", - }); - searchFilterCmds: OV = mobx.observable.box(true, { - name: "historyview-filtercmds", - }); - nextRawOffset: number = 0; - curRawOffset: number = 0; - - historyItemLines: LineType[] = []; - historyItemCmds: CmdDataType[] = []; - - specialLineContainer: SpecialLineContainer; - - closeView(): void { - GlobalModel.showSessionView(); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - } - - getLineById(lineId: string): LineType { - if (isBlank(lineId)) { - return null; - } - for (const line of this.historyItemLines) { - if (line.lineid == lineId) { - return line; - } - } - return null; - } - - getCmdById(lineId: string): Cmd { - if (isBlank(lineId)) { - return null; - } - for (const cmd of this.historyItemCmds) { - if (cmd.lineid == lineId) { - return new Cmd(cmd); - } - } - return null; - } - - getHistoryItemById(historyId: string): HistoryItem { - if (isBlank(historyId)) { - return null; - } - for (const hitem of this.items) { - if (hitem.historyid == historyId) { - return hitem; - } - } - return null; - } - - setActiveItem(historyId: string) { - if (this.activeItem.get() == historyId) { - return; - } - let hitem = this.getHistoryItemById(historyId); - mobx.action(() => { - if (hitem == null) { - this.activeItem.set(null); - this.specialLineContainer = null; - } else { - this.activeItem.set(hitem.historyid); - let width = termWidthFromCols(80, GlobalModel.termFontSize.get()); - let height = termHeightFromRows(25, GlobalModel.termFontSize.get(), 25); - this.specialLineContainer = new SpecialLineContainer( - this, - { width, height }, - false, - appconst.LineContainer_History - ); - } - })(); - } - - doSelectedDelete(): void { - if (!this.deleteActive.get()) { - mobx.action(() => { - this.deleteActive.set(true); - })(); - setTimeout(this.clearActiveDelete, 2000); - return; - } - let prtn = GlobalModel.showAlert({ - message: "Deleting lines from history also deletes their content from your workspaces.", - confirm: true, - }); - prtn.then((result) => { - if (!result) { - return; - } - if (result) { - this._deleteSelected(); - } - }); - } - - _deleteSelected(): void { - let lineIds = Array.from(this.selectedItems.keys()); - let prtn = GlobalCommandRunner.historyPurgeLines(lineIds); - prtn.then((result: CommandRtnType) => { - if (!result.success) { - GlobalModel.showAlert({ message: "Error removing history lines." }); - } - }); - let params = this._getSearchParams(); - GlobalCommandRunner.historyView(params); - } - - @boundMethod - clearActiveDelete(): void { - mobx.action(() => { - this.deleteActive.set(false); - })(); - } - - _getSearchParams(newOffset?: number, newRawOffset?: number): HistorySearchParams { - let offset = newOffset ?? this.offset.get(); - let rawOffset = newRawOffset ?? this.curRawOffset; - let opts: HistorySearchParams = { - offset: offset, - rawOffset: rawOffset, - searchText: this.activeSearchText, - searchSessionId: this.searchSessionId.get(), - searchRemoteId: this.searchRemoteId.get(), - }; - if (!this.searchShowMeta.get()) { - opts.noMeta = true; - } - if (this.searchFromDate.get() != null) { - let fromDate = this.searchFromDate.get(); - let fromTs = dayjs(fromDate, "YYYY-MM-DD").valueOf(); - let d = new Date(fromTs); - d.setDate(d.getDate() + 1); - let ts = d.getTime() - 1; - opts.fromTs = ts; - } - if (this.searchFilterCmds.get()) { - opts.filterCmds = true; - } - return opts; - } - - reSearch(): void { - this.setActiveItem(null); - GlobalCommandRunner.historyView(this._getSearchParams()); - } - - resetAllFilters(): void { - mobx.action(() => { - this.activeSearchText = ""; - this.searchText.set(""); - this.searchSessionId.set(null); - this.searchRemoteId.set(null); - this.searchFromDate.set(null); - this.searchShowMeta.set(true); - this.searchFilterCmds.set(true); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - setFromDate(fromDate: string): void { - if (this.searchFromDate.get() == fromDate) { - return; - } - mobx.action(() => { - this.searchFromDate.set(fromDate); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - setSearchFilterCmds(filter: boolean): void { - if (this.searchFilterCmds.get() == filter) { - return; - } - mobx.action(() => { - this.searchFilterCmds.set(filter); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - setSearchShowMeta(show: boolean): void { - if (this.searchShowMeta.get() == show) { - return; - } - mobx.action(() => { - this.searchShowMeta.set(show); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - setSearchSessionId(sessionId: string): void { - if (this.searchSessionId.get() == sessionId) { - return; - } - mobx.action(() => { - this.searchSessionId.set(sessionId); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - setSearchRemoteId(remoteId: string): void { - if (this.searchRemoteId.get() == remoteId) { - return; - } - mobx.action(() => { - this.searchRemoteId.set(remoteId); - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - goPrev(): void { - let offset = this.offset.get(); - offset = offset - HistoryPageSize; - if (offset < 0) { - offset = 0; - } - let params = this._getSearchParams(offset, 0); - GlobalCommandRunner.historyView(params); - } - - goNext(): void { - let offset = this.offset.get(); - offset += HistoryPageSize; - let params = this._getSearchParams(offset, this.nextRawOffset ?? 0); - GlobalCommandRunner.historyView(params); - } - - submitSearch(): void { - mobx.action(() => { - this.hasMore.set(false); - this.items.replace([]); - this.activeSearchText = this.searchText.get(); - this.historyItemLines = []; - this.historyItemCmds = []; - })(); - GlobalCommandRunner.historyView(this._getSearchParams(0, 0)); - } - - handleDocKeyDown(e: any): void { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if (checkKeyPressed(waveEvent, "Escape")) { - e.preventDefault(); - this.closeView(); - return; - } - } - - showHistoryView(data: HistoryViewDataType): void { - mobx.action(() => { - GlobalModel.activeMainView.set("history"); - this.hasMore.set(data.hasmore); - this.items.replace(data.items || []); - this.offset.set(data.offset); - this.nextRawOffset = data.nextrawoffset; - this.curRawOffset = data.rawoffset; - this.historyItemLines = data.lines ?? []; - this.historyItemCmds = data.cmds ?? []; - this.selectedItems.clear(); - })(); - } -} - -class ConnectionsViewModel { - closeView(): void { - GlobalModel.showSessionView(); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - } - - showConnectionsView(): void { - mobx.action(() => { - GlobalModel.activeMainView.set("connections"); - })(); - } -} - -class ClientSettingsViewModel { - closeView(): void { - GlobalModel.showSessionView(); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - } - - showClientSettingsView(): void { - mobx.action(() => { - GlobalModel.activeMainView.set("clientsettings"); - })(); - } -} -class MainSidebarModel { - tempWidth: OV = mobx.observable.box(null, { - name: "MainSidebarModel-tempWidth", - }); - tempCollapsed: OV = mobx.observable.box(null, { - name: "MainSidebarModel-tempCollapsed", - }); - isDragging: OV = mobx.observable.box(false, { - name: "MainSidebarModel-isDragging", - }); - - setTempWidthAndTempCollapsed(newWidth: number, newCollapsed: boolean): void { - const width = Math.max(MagicLayout.MainSidebarMinWidth, Math.min(newWidth, MagicLayout.MainSidebarMaxWidth)); - - mobx.action(() => { - this.tempWidth.set(width); - this.tempCollapsed.set(newCollapsed); - })(); - } - - /** - * Gets the intended width for the sidebar. If the sidebar is being dragged, returns the tempWidth. If the sidebar is collapsed, returns the default width. - * @param ignoreCollapse If true, returns the persisted width even if the sidebar is collapsed. - * @returns The intended width for the sidebar or the default width if the sidebar is collapsed. Can be overridden using ignoreCollapse. - */ - getWidth(ignoreCollapse: boolean = false): number { - const clientData = GlobalModel.clientData.get(); - let width = clientData?.clientopts?.mainsidebar?.width ?? MagicLayout.MainSidebarDefaultWidth; - if (this.isDragging.get()) { - if (this.tempWidth.get() == null && width == null) { - return MagicLayout.MainSidebarDefaultWidth; - } - if (this.tempWidth.get() == null) { - return width; - } - return this.tempWidth.get(); - } - // Set by CLI and collapsed - if (this.getCollapsed()) { - if (ignoreCollapse) { - return width; - } else { - return MagicLayout.MainSidebarMinWidth; - } - } else { - if (width <= MagicLayout.MainSidebarMinWidth) { - width = MagicLayout.MainSidebarDefaultWidth; - } - const snapPoint = MagicLayout.MainSidebarMinWidth + MagicLayout.MainSidebarSnapThreshold; - if (width < snapPoint || width > MagicLayout.MainSidebarMaxWidth) { - width = MagicLayout.MainSidebarDefaultWidth; - } - } - return width; - } - - getCollapsed(): boolean { - const clientData = GlobalModel.clientData.get(); - const collapsed = clientData?.clientopts?.mainsidebar?.collapsed; - if (this.isDragging.get()) { - if (this.tempCollapsed.get() == null && collapsed == null) { - return false; - } - if (this.tempCollapsed.get() == null) { - return collapsed; - } - return this.tempCollapsed.get(); - } - return collapsed; - } -} - -class BookmarksModel { - bookmarks: OArr = mobx.observable.array([], { - name: "Bookmarks", - }); - activeBookmark: OV = mobx.observable.box(null, { - name: "activeBookmark", - }); - editingBookmark: OV = mobx.observable.box(null, { - name: "editingBookmark", - }); - pendingDelete: OV = mobx.observable.box(null, { - name: "pendingDelete", - }); - copiedIndicator: OV = mobx.observable.box(null, { - name: "copiedIndicator", - }); - - tempDesc: OV = mobx.observable.box("", { - name: "bookmarkEdit-tempDesc", - }); - tempCmd: OV = mobx.observable.box("", { - name: "bookmarkEdit-tempCmd", - }); - - showBookmarksView(bmArr: BookmarkType[], selectedBookmarkId: string): void { - bmArr = bmArr ?? []; - mobx.action(() => { - this.reset(); - GlobalModel.activeMainView.set("bookmarks"); - this.bookmarks.replace(bmArr); - if (selectedBookmarkId != null) { - this.selectBookmark(selectedBookmarkId); - } - if (this.activeBookmark.get() == null && bmArr.length > 0) { - this.activeBookmark.set(bmArr[0].bookmarkid); - } - })(); - } - - reset(): void { - mobx.action(() => { - this.activeBookmark.set(null); - this.editingBookmark.set(null); - this.pendingDelete.set(null); - this.tempDesc.set(""); - this.tempCmd.set(""); - })(); - } - - closeView(): void { - GlobalModel.showSessionView(); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - } - - @boundMethod - clearPendingDelete(): void { - mobx.action(() => this.pendingDelete.set(null))(); - } - - useBookmark(bookmarkId: string): void { - let bm = this.getBookmark(bookmarkId); - if (bm == null) { - return; - } - mobx.action(() => { - this.reset(); - GlobalModel.showSessionView(); - GlobalModel.inputModel.setCurLine(bm.cmdstr); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - })(); - } - - selectBookmark(bookmarkId: string): void { - let bm = this.getBookmark(bookmarkId); - if (bm == null) { - return; - } - if (this.activeBookmark.get() == bookmarkId) { - return; - } - mobx.action(() => { - this.cancelEdit(); - this.activeBookmark.set(bookmarkId); - })(); - } - - cancelEdit(): void { - mobx.action(() => { - this.pendingDelete.set(null); - this.editingBookmark.set(null); - this.tempDesc.set(""); - this.tempCmd.set(""); - })(); - } - - confirmEdit(): void { - if (this.editingBookmark.get() == null) { - return; - } - let bm = this.getBookmark(this.editingBookmark.get()); - mobx.action(() => { - this.editingBookmark.set(null); - bm.description = this.tempDesc.get(); - bm.cmdstr = this.tempCmd.get(); - this.tempDesc.set(""); - this.tempCmd.set(""); - })(); - GlobalCommandRunner.editBookmark(bm.bookmarkid, bm.description, bm.cmdstr); - } - - handleDeleteBookmark(bookmarkId: string): void { - if (this.pendingDelete.get() == null || this.pendingDelete.get() != this.activeBookmark.get()) { - mobx.action(() => this.pendingDelete.set(this.activeBookmark.get()))(); - setTimeout(this.clearPendingDelete, 2000); - return; - } - GlobalCommandRunner.deleteBookmark(bookmarkId); - this.clearPendingDelete(); - } - - getBookmark(bookmarkId: string): BookmarkType { - if (bookmarkId == null) { - return null; - } - for (const bm of this.bookmarks) { - if (bm.bookmarkid == bookmarkId) { - return bm; - } - } - return null; - } - - getBookmarkPos(bookmarkId: string): number { - if (bookmarkId == null) { - return -1; - } - for (let i = 0; i < this.bookmarks.length; i++) { - let bm = this.bookmarks[i]; - if (bm.bookmarkid == bookmarkId) { - return i; - } - } - return -1; - } - - getActiveBookmark(): BookmarkType { - let activeBookmarkId = this.activeBookmark.get(); - return this.getBookmark(activeBookmarkId); - } - - handleEditBookmark(bookmarkId: string): void { - let bm = this.getBookmark(bookmarkId); - if (bm == null) { - return; - } - mobx.action(() => { - this.pendingDelete.set(null); - this.activeBookmark.set(bookmarkId); - this.editingBookmark.set(bookmarkId); - this.tempDesc.set(bm.description ?? ""); - this.tempCmd.set(bm.cmdstr ?? ""); - })(); - } - - handleCopyBookmark(bookmarkId: string): void { - let bm = this.getBookmark(bookmarkId); - if (bm == null) { - return; - } - navigator.clipboard.writeText(bm.cmdstr); - mobx.action(() => { - this.copiedIndicator.set(bm.bookmarkid); - })(); - setTimeout(() => { - mobx.action(() => { - this.copiedIndicator.set(null); - })(); - }, 600); - } - - mergeBookmarks(bmArr: BookmarkType[]): void { - mobx.action(() => { - genMergeSimpleData( - this.bookmarks, - bmArr, - (bm: BookmarkType) => bm.bookmarkid, - (bm: BookmarkType) => sprintf("%05d", bm.orderidx) - ); - })(); - } - - handleDocKeyDown(e: any): void { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if (checkKeyPressed(waveEvent, "Escape")) { - e.preventDefault(); - if (this.editingBookmark.get() != null) { - this.cancelEdit(); - return; - } - this.closeView(); - return; - } - if (this.editingBookmark.get() != null) { - return; - } - if (checkKeyPressed(waveEvent, "Backspace") || checkKeyPressed(waveEvent, "Delete")) { - if (this.activeBookmark.get() == null) { - return; - } - e.preventDefault(); - this.handleDeleteBookmark(this.activeBookmark.get()); - return; - } - - if ( - checkKeyPressed(waveEvent, "ArrowUp") || - checkKeyPressed(waveEvent, "ArrowDown") || - checkKeyPressed(waveEvent, "PageUp") || - checkKeyPressed(waveEvent, "PageDown") - ) { - e.preventDefault(); - if (this.bookmarks.length == 0) { - return; - } - let newPos = 0; // if active is null, then newPos will be 0 (select the first) - if (this.activeBookmark.get() != null) { - let amtMap = { ArrowUp: -1, ArrowDown: 1, PageUp: -10, PageDown: 10 }; - let amt = amtMap[e.code]; - let curIdx = this.getBookmarkPos(this.activeBookmark.get()); - newPos = curIdx + amt; - if (newPos < 0) { - newPos = 0; - } - if (newPos >= this.bookmarks.length) { - newPos = this.bookmarks.length - 1; - } - } - let bm = this.bookmarks[newPos]; - mobx.action(() => { - this.activeBookmark.set(bm.bookmarkid); - })(); - return; - } - if (checkKeyPressed(waveEvent, "Enter")) { - if (this.activeBookmark.get() == null) { - return; - } - this.useBookmark(this.activeBookmark.get()); - return; - } - if (checkKeyPressed(waveEvent, "e")) { - if (this.activeBookmark.get() == null) { - return; - } - e.preventDefault(); - this.handleEditBookmark(this.activeBookmark.get()); - return; - } - if (checkKeyPressed(waveEvent, "c")) { - if (this.activeBookmark.get() == null) { - return; - } - e.preventDefault(); - this.handleCopyBookmark(this.activeBookmark.get()); - } - } -} - -class PluginsModel { - selectedPlugin: OV = mobx.observable.box(null, { name: "selectedPlugin" }); - - showPluginsView(): void { - PluginModel.loadAllPluginResources(); - mobx.action(() => { - this.reset(); - GlobalModel.activeMainView.set("plugins"); - const allPlugins = PluginModel.allPlugins(); - this.selectedPlugin.set(allPlugins.length > 0 ? allPlugins[0] : null); - })(); - } - - setSelectedPlugin(plugin: RendererPluginType): void { - mobx.action(() => { - this.selectedPlugin.set(plugin); - })(); - } - - reset(): void { - mobx.action(() => { - this.selectedPlugin.set(null); - })(); - } - - closeView(): void { - GlobalModel.showSessionView(); - setTimeout(() => GlobalModel.inputModel.giveFocus(), 50); - } -} - -class RemotesModalModel { - openState: OV = mobx.observable.box(false, { - name: "RemotesModalModel-isOpen", - }); - onlyAddNewRemote: OV = mobx.observable.box(false, { - name: "RemotesModalModel-onlyAddNewRemote", - }); - selectedRemoteId: OV = mobx.observable.box(null, { - name: "RemotesModalModel-selectedRemoteId", - }); - remoteTermWrap: TermWrap; - remoteTermWrapFocus: OV = mobx.observable.box(false, { - name: "RemotesModalModel-remoteTermWrapFocus", - }); - showNoInputMsg: OV = mobx.observable.box(false, { - name: "RemotesModel-showNoInputMg", - }); - showNoInputTimeoutId: any = null; - remoteEdit: OV = mobx.observable.box(null, { - name: "RemoteModal-remoteEdit", - }); - - openModal(remoteId?: string): void { - if (remoteId == null) { - let ri = GlobalModel.getCurRemoteInstance(); - if (ri != null) { - remoteId = ri.remoteid; - } else { - let localRemote = GlobalModel.getLocalRemote(); - if (localRemote != null) { - remoteId = localRemote.remoteid; - } - } - } - mobx.action(() => { - this.openState.set(true); - this.selectedRemoteId.set(remoteId); - this.remoteEdit.set(null); - })(); - } - - deSelectRemote(): void { - mobx.action(() => { - this.selectedRemoteId.set(null); - this.remoteEdit.set(null); - })(); - } - - openModalForEdit(redit: RemoteEditType, onlyAddNewRemote: boolean): void { - mobx.action(() => { - this.onlyAddNewRemote.set(onlyAddNewRemote); - this.openState.set(true); - this.selectedRemoteId.set(redit.remoteid); - this.remoteEdit.set(redit); - })(); - } - - selectRemote(remoteId: string): void { - if (this.selectedRemoteId.get() == remoteId) { - return; - } - mobx.action(() => { - this.selectedRemoteId.set(remoteId); - this.remoteEdit.set(null); - })(); - } - - @boundMethod - startEditAuth(): void { - let remoteId = this.selectedRemoteId.get(); - if (remoteId != null) { - GlobalCommandRunner.openEditRemote(remoteId); - } - } - - @boundMethod - cancelEditAuth(): void { - mobx.action(() => { - this.remoteEdit.set(null); - if (this.onlyAddNewRemote.get()) { - this.onlyAddNewRemote.set(false); - this.openState.set(false); - return; - } - if (this.selectedRemoteId.get() == null) { - this.openModal(); - } - })(); - } - - isOpen(): boolean { - return this.openState.get(); - } - - isAuthEditMode(): boolean { - return this.remoteEdit.get() != null; - } - - closeModal(): void { - if (!this.openState.get()) { - return; - } - mobx.action(() => { - this.openState.set(false); - this.selectedRemoteId.set(null); - this.remoteEdit.set(null); - this.onlyAddNewRemote.set(false); - })(); - setTimeout(() => GlobalModel.refocus(), 10); - } - - disposeTerm(): void { - if (this.remoteTermWrap == null) { - return; - } - this.remoteTermWrap.dispose(); - this.remoteTermWrap = null; - mobx.action(() => { - this.remoteTermWrapFocus.set(false); - })(); - } - - receiveData(remoteId: string, ptyPos: number, ptyData: Uint8Array, reason?: string) { - if (this.remoteTermWrap == null) { - return; - } - if (this.remoteTermWrap.getContextRemoteId() != remoteId) { - return; - } - this.remoteTermWrap.receiveData(ptyPos, ptyData); - } - - @boundMethod - setRemoteTermWrapFocus(focus: boolean): void { - mobx.action(() => { - this.remoteTermWrapFocus.set(focus); - })(); - } - - @boundMethod - setShowNoInputMsg(val: boolean) { - mobx.action(() => { - if (this.showNoInputTimeoutId != null) { - clearTimeout(this.showNoInputTimeoutId); - this.showNoInputTimeoutId = null; - } - if (val) { - this.showNoInputMsg.set(true); - this.showNoInputTimeoutId = setTimeout(() => this.setShowNoInputMsg(false), 2000); - } else { - this.showNoInputMsg.set(false); - } - })(); - } - - @boundMethod - termKeyHandler(remoteId: string, event: any, termWrap: TermWrap): void { - let remote = GlobalModel.getRemote(remoteId); - if (remote == null) { - return; - } - if (remote.status != "connecting" && remote.installstatus != "connecting") { - this.setShowNoInputMsg(true); - return; - } - let inputPacket: RemoteInputPacketType = { - type: "remoteinput", - remoteid: remoteId, - inputdata64: stringToBase64(event.key), - }; - GlobalModel.sendInputPacket(inputPacket); - } - - createTermWrap(elem: HTMLElement): void { - this.disposeTerm(); - let remoteId = this.selectedRemoteId.get(); - if (remoteId == null) { - return; - } - let termOpts = { - rows: RemotePtyRows, - cols: RemotePtyCols, - flexrows: false, - maxptysize: 64 * 1024, - }; - let termWrap = new TermWrap(elem, { - termContext: { remoteId: remoteId }, - usedRows: RemotePtyRows, - termOpts: termOpts, - winSize: null, - keyHandler: (e, termWrap) => { - this.termKeyHandler(remoteId, e, termWrap); - }, - focusHandler: this.setRemoteTermWrapFocus.bind(this), - isRunning: true, - fontSize: GlobalModel.termFontSize.get(), - ptyDataSource: getTermPtyData, - onUpdateContentHeight: null, - }); - this.remoteTermWrap = termWrap; - } -} - -class RemotesModel { - selectedRemoteId: OV = mobx.observable.box(null, { - name: "RemotesModel-selectedRemoteId", - }); - remoteTermWrap: TermWrap = null; - remoteTermWrapFocus: OV = mobx.observable.box(false, { - name: "RemotesModel-remoteTermWrapFocus", - }); - showNoInputMsg: OV = mobx.observable.box(false, { - name: "RemotesModel-showNoInputMg", - }); - showNoInputTimeoutId: any = null; - remoteEdit: OV = mobx.observable.box(null, { - name: "RemotesModel-remoteEdit", - }); - recentConnAddedState: OV = mobx.observable.box(false, { - name: "RemotesModel-recentlyAdded", - }); - - get recentConnAdded(): boolean { - return this.recentConnAddedState.get(); - } - - @boundMethod - setRecentConnAdded(value: boolean) { - mobx.action(() => { - this.recentConnAddedState.set(value); - })(); - } - - deSelectRemote(): void { - mobx.action(() => { - this.selectedRemoteId.set(null); - this.remoteEdit.set(null); - })(); - } - - openReadModal(remoteId: string): void { - mobx.action(() => { - this.setRecentConnAdded(false); - this.selectedRemoteId.set(remoteId); - this.remoteEdit.set(null); - GlobalModel.modalsModel.pushModal(appconst.VIEW_REMOTE); - })(); - } - - openAddModal(redit: RemoteEditType): void { - mobx.action(() => { - this.remoteEdit.set(redit); - GlobalModel.modalsModel.pushModal(appconst.CREATE_REMOTE); - })(); - } - - openEditModal(redit?: RemoteEditType): void { - mobx.action(() => { - this.selectedRemoteId.set(redit?.remoteid); - this.remoteEdit.set(redit); - GlobalModel.modalsModel.pushModal(appconst.EDIT_REMOTE); - })(); - } - - selectRemote(remoteId: string): void { - if (this.selectedRemoteId.get() == remoteId) { - return; - } - mobx.action(() => { - this.selectedRemoteId.set(remoteId); - this.remoteEdit.set(null); - })(); - } - - @boundMethod - startEditAuth(): void { - let remoteId = this.selectedRemoteId.get(); - if (remoteId != null) { - GlobalCommandRunner.openEditRemote(remoteId); - } - } - - isAuthEditMode(): boolean { - return this.remoteEdit.get() != null; - } - - @boundMethod - closeModal(): void { - mobx.action(() => { - GlobalModel.modalsModel.popModal(); - })(); - setTimeout(() => GlobalModel.refocus(), 10); - } - - disposeTerm(): void { - if (this.remoteTermWrap == null) { - return; - } - this.remoteTermWrap.dispose(); - this.remoteTermWrap = null; - mobx.action(() => { - this.remoteTermWrapFocus.set(false); - })(); - } - - receiveData(remoteId: string, ptyPos: number, ptyData: Uint8Array, reason?: string) { - if (this.remoteTermWrap == null) { - return; - } - if (this.remoteTermWrap.getContextRemoteId() != remoteId) { - return; - } - this.remoteTermWrap.receiveData(ptyPos, ptyData); - } - - @boundMethod - setRemoteTermWrapFocus(focus: boolean): void { - mobx.action(() => { - this.remoteTermWrapFocus.set(focus); - })(); - } - - @boundMethod - setShowNoInputMsg(val: boolean) { - mobx.action(() => { - if (this.showNoInputTimeoutId != null) { - clearTimeout(this.showNoInputTimeoutId); - this.showNoInputTimeoutId = null; - } - if (val) { - this.showNoInputMsg.set(true); - this.showNoInputTimeoutId = setTimeout(() => this.setShowNoInputMsg(false), 2000); - } else { - this.showNoInputMsg.set(false); - } - })(); - } - - @boundMethod - termKeyHandler(remoteId: string, event: any, termWrap: TermWrap): void { - let remote = GlobalModel.getRemote(remoteId); - if (remote == null) { - return; - } - if (remote.status != "connecting" && remote.installstatus != "connecting") { - this.setShowNoInputMsg(true); - return; - } - let inputPacket: RemoteInputPacketType = { - type: "remoteinput", - remoteid: remoteId, - inputdata64: stringToBase64(event.key), - }; - GlobalModel.sendInputPacket(inputPacket); - } - - createTermWrap(elem: HTMLElement): void { - this.disposeTerm(); - let remoteId = this.selectedRemoteId.get(); - if (remoteId == null) { - return; - } - let termOpts = { - rows: RemotePtyRows, - cols: RemotePtyCols, - flexrows: false, - maxptysize: 64 * 1024, - }; - let termWrap = new TermWrap(elem, { - termContext: { remoteId: remoteId }, - usedRows: RemotePtyRows, - termOpts: termOpts, - winSize: null, - keyHandler: (e, termWrap) => { - this.termKeyHandler(remoteId, e, termWrap); - }, - focusHandler: this.setRemoteTermWrapFocus.bind(this), - isRunning: true, - fontSize: GlobalModel.termFontSize.get(), - ptyDataSource: getTermPtyData, - onUpdateContentHeight: null, - }); - this.remoteTermWrap = termWrap; - } -} - -class ModalsModel { - store: OArr = mobx.observable.array([], { name: "ModalsModel-store", deep: false }); - - pushModal(modalId: string, props?: any) { - const modalFactory = modalsRegistry[modalId]; - - if (modalFactory && !this.store.some((modal) => modal.id === modalId)) { - mobx.action(() => { - this.store.push({ id: modalId, component: modalFactory, uniqueKey: uuidv4(), props: props }); - })(); - } - } - - popModal() { - mobx.action(() => { - this.store.pop(); - })(); - } -} - -class Model { - clientId: string; - activeSessionId: OV = mobx.observable.box(null, { - name: "activeSessionId", - }); - sessionListLoaded: OV = mobx.observable.box(false, { - name: "sessionListLoaded", - }); - sessionList: OArr = mobx.observable.array([], { - name: "SessionList", - deep: false, - }); - screenMap: OMap = mobx.observable.map({}, { name: "ScreenMap", deep: false }); - ws: WSControl; - remotes: OArr = mobx.observable.array([], { - name: "remotes", - deep: false, - }); - remotesLoaded: OV = mobx.observable.box(false, { - name: "remotesLoaded", - }); - screenLines: OMap = mobx.observable.map({}, { name: "screenLines", deep: false }); // key = "sessionid/screenid" (screenlines) - termUsedRowsCache: Record = {}; // key = "screenid/lineid" - debugCmds: number = 0; - debugScreen: OV = mobx.observable.box(false); - waveSrvRunning: OV; - authKey: string; - isDev: boolean; - platform: string; - activeMainView: OV< - "plugins" | "session" | "history" | "bookmarks" | "webshare" | "connections" | "clientsettings" - > = mobx.observable.box("session", { - name: "activeMainView", - }); - termFontSize: CV; - alertMessage: OV = mobx.observable.box(null, { - name: "alertMessage", - }); - alertPromiseResolver: (result: boolean) => void; - aboutModalOpen: OV = mobx.observable.box(false, { - name: "aboutModalOpen", - }); - screenSettingsModal: OV<{ sessionId: string; screenId: string }> = mobx.observable.box(null, { - name: "screenSettingsModal", - }); - sessionSettingsModal: OV = mobx.observable.box(null, { - name: "sessionSettingsModal", - }); - clientSettingsModal: OV = mobx.observable.box(false, { - name: "clientSettingsModal", - }); - lineSettingsModal: OV = mobx.observable.box(null, { - name: "lineSettingsModal", - }); // linenum - devicePixelRatio: OV = mobx.observable.box(window.devicePixelRatio, { - name: "devicePixelRatio", - }); - remotesModalModel: RemotesModalModel; - remotesModel: RemotesModel; - - inputModel: InputModel; - pluginsModel: PluginsModel; - bookmarksModel: BookmarksModel; - historyViewModel: HistoryViewModel; - connectionViewModel: ConnectionsViewModel; - clientSettingsViewModel: ClientSettingsViewModel; - modalsModel: ModalsModel; - mainSidebarModel: MainSidebarModel; - clientData: OV = mobx.observable.box(null, { - name: "clientData", - }); - showLinks: OV = mobx.observable.box(true, { - name: "model-showLinks", - }); - packetSeqNum: number = 0; - - constructor() { - this.clientId = getApi().getId(); - this.isDev = getApi().getIsDev(); - this.authKey = getApi().getAuthKey(); - this.ws = new WSControl(this.getBaseWsHostPort(), this.clientId, this.authKey, (message: any) => { - let interactive = message?.interactive ?? false; - this.runUpdate(message, interactive); - }); - this.ws.reconnect(); - this.inputModel = new InputModel(); - this.pluginsModel = new PluginsModel(); - this.bookmarksModel = new BookmarksModel(); - this.historyViewModel = new HistoryViewModel(); - this.connectionViewModel = new ConnectionsViewModel(); - this.clientSettingsViewModel = new ClientSettingsViewModel(); - this.remotesModalModel = new RemotesModalModel(); - this.remotesModel = new RemotesModel(); - this.modalsModel = new ModalsModel(); - this.mainSidebarModel = new MainSidebarModel(); - let isWaveSrvRunning = getApi().getWaveSrvStatus(); - this.waveSrvRunning = mobx.observable.box(isWaveSrvRunning, { - name: "model-wavesrv-running", - }); - this.platform = this.getPlatform(); - this.termFontSize = mobx.computed(() => { - let cdata = this.clientData.get(); - if (cdata == null || cdata.feopts == null || cdata.feopts.termfontsize == null) { - return DefaultTermFontSize; - } - let fontSize = Math.ceil(cdata.feopts.termfontsize); - if (fontSize < MinFontSize) { - return MinFontSize; - } - if (fontSize > MaxFontSize) { - return MaxFontSize; - } - return fontSize; - }); - getApi().onTCmd(this.onTCmd.bind(this)); - getApi().onICmd(this.onICmd.bind(this)); - getApi().onLCmd(this.onLCmd.bind(this)); - getApi().onHCmd(this.onHCmd.bind(this)); - getApi().onPCmd(this.onPCmd.bind(this)); - getApi().onWCmd(this.onWCmd.bind(this)); - getApi().onRCmd(this.onRCmd.bind(this)); - getApi().onZoomChanged(this.onZoomChanged.bind(this)); - getApi().onMenuItemAbout(this.onMenuItemAbout.bind(this)); - getApi().onMetaArrowUp(this.onMetaArrowUp.bind(this)); - getApi().onMetaArrowDown(this.onMetaArrowDown.bind(this)); - getApi().onMetaPageUp(this.onMetaPageUp.bind(this)); - getApi().onMetaPageDown(this.onMetaPageDown.bind(this)); - getApi().onBracketCmd(this.onBracketCmd.bind(this)); - getApi().onDigitCmd(this.onDigitCmd.bind(this)); - getApi().onWaveSrvStatusChange(this.onWaveSrvStatusChange.bind(this)); - document.addEventListener("keydown", this.docKeyDownHandler.bind(this)); - document.addEventListener("selectionchange", this.docSelectionChangeHandler.bind(this)); - setTimeout(() => this.getClientDataLoop(1), 10); - } - - getNextPacketSeqNum(): number { - this.packetSeqNum++; - return this.packetSeqNum; - } - - getPlatform(): string { - if (this.platform != null) { - return this.platform; - } - this.platform = getApi().getPlatform(); - setKeyUtilPlatform(this.platform); - return this.platform; - } - - testGlobalModel() { - return ""; - } - - needsTos(): boolean { - let cdata = this.clientData.get(); - if (cdata == null) { - return false; - } - return cdata.clientopts == null || !cdata.clientopts.acceptedtos; - } - - refreshClient(): void { - getApi().reloadWindow(); - } - - /** - * Opens a new default browser window to the given url - * @param {string} url The url to open - */ - openExternalLink(url: string): void { - console.log("opening external link: " + url); - getApi().openExternalLink(url); - console.log("finished opening external link"); - } - - refocus() { - // givefocus() give back focus to cmd or input - let activeScreen = this.getActiveScreen(); - if (screen == null) { - return; - } - activeScreen.giveFocus(); - } - - getWebSharedScreens(): Screen[] { - let rtn: Screen[] = []; - for (let screen of this.screenMap.values()) { - if (screen.shareMode.get() == "web") { - rtn.push(screen); - } - } - return rtn; - } - - getHasClientStop(): boolean { - if (this.clientData.get() == null) { - return true; - } - let cdata = this.clientData.get(); - if (cdata.cmdstoretype == "session") { - return true; - } - return false; - } - - showAlert(alertMessage: AlertMessageType): Promise { - if (alertMessage.confirmflag != null) { - let cdata = GlobalModel.clientData.get(); - let noConfirm = cdata.clientopts?.confirmflags?.[alertMessage.confirmflag]; - if (noConfirm) { - return Promise.resolve(true); - } - } - mobx.action(() => { - this.alertMessage.set(alertMessage); - GlobalModel.modalsModel.pushModal(appconst.ALERT); - })(); - let prtn = new Promise((resolve, reject) => { - this.alertPromiseResolver = resolve; - }); - return prtn; - } - - cancelAlert(): void { - mobx.action(() => { - this.alertMessage.set(null); - GlobalModel.modalsModel.popModal(); - })(); - if (this.alertPromiseResolver != null) { - this.alertPromiseResolver(false); - this.alertPromiseResolver = null; - } - } - - confirmAlert(): void { - mobx.action(() => { - this.alertMessage.set(null); - GlobalModel.modalsModel.popModal(); - })(); - if (this.alertPromiseResolver != null) { - this.alertPromiseResolver(true); - this.alertPromiseResolver = null; - } - } - - showSessionView(): void { - mobx.action(() => { - this.activeMainView.set("session"); - })(); - } - - showWebShareView(): void { - mobx.action(() => { - this.activeMainView.set("webshare"); - })(); - } - - getBaseHostPort(): string { - if (this.isDev) { - return DevServerEndpoint; - } - return ProdServerEndpoint; - } - - setTermFontSize(fontSize: number) { - if (fontSize < MinFontSize) { - fontSize = MinFontSize; - } - if (fontSize > MaxFontSize) { - fontSize = MaxFontSize; - } - mobx.action(() => { - this.termFontSize.set(fontSize); - })(); - } - - getBaseWsHostPort(): string { - if (this.isDev) { - return DevServerWsEndpoint; - } - return ProdServerWsEndpoint; - } - - getFetchHeaders(): Record { - return { - "x-authkey": this.authKey, - }; - } - - docSelectionChangeHandler(e: any) { - // nothing for now - } - - docKeyDownHandler(e: KeyboardEvent) { - let waveEvent = adaptFromReactOrNativeKeyEvent(e); - if (isModKeyPress(e)) { - return; - } - if (this.alertMessage.get() != null) { - if (checkKeyPressed(waveEvent, "Escape")) { - e.preventDefault(); - this.cancelAlert(); - return; - } - if (checkKeyPressed(waveEvent, "Enter")) { - e.preventDefault(); - this.confirmAlert(); - return; - } - return; - } - if (this.activeMainView.get() == "bookmarks") { - this.bookmarksModel.handleDocKeyDown(e); - return; - } - if (this.activeMainView.get() == "history") { - this.historyViewModel.handleDocKeyDown(e); - return; - } - if (this.activeMainView.get() == "connections") { - this.historyViewModel.handleDocKeyDown(e); - return; - } - if (this.activeMainView.get() == "clientsettings") { - this.historyViewModel.handleDocKeyDown(e); - return; - } - if (checkKeyPressed(waveEvent, "Escape")) { - e.preventDefault(); - if (this.activeMainView.get() == "webshare") { - this.showSessionView(); - return; - } - if (this.clearModals()) { - return; - } - let inputModel = this.inputModel; - inputModel.toggleInfoMsg(); - if (inputModel.inputMode.get() != null) { - inputModel.resetInputMode(); - } - return; - } - if (checkKeyPressed(waveEvent, "Cmd:b")) { - e.preventDefault(); - GlobalCommandRunner.bookmarksView(); - } - if (this.activeMainView.get() == "session" && checkKeyPressed(waveEvent, "Cmd:Ctrl:s")) { - e.preventDefault(); - let activeScreen = this.getActiveScreen(); - if (activeScreen != null) { - let isSidebarOpen = activeScreen.isSidebarOpen(); - if (isSidebarOpen) { - GlobalCommandRunner.screenSidebarClose(); - } else { - GlobalCommandRunner.screenSidebarOpen(); - } - } - } - if (checkKeyPressed(waveEvent, "Cmd:d")) { - let ranDelete = this.deleteActiveLine(); - if (ranDelete) { - e.preventDefault(); - } - } - } - - deleteActiveLine(): boolean { - let activeScreen = this.getActiveScreen(); - if (activeScreen == null || activeScreen.getFocusType() != "cmd") { - return false; - } - let selectedLine = activeScreen.selectedLine.get(); - if (selectedLine == null || selectedLine <= 0) { - return false; - } - let line = activeScreen.getLineByNum(selectedLine); - if (line == null) { - return false; - } - let cmd = activeScreen.getCmd(line); - if (cmd != null) { - if (cmd.isRunning()) { - let info: T.InfoType = { infomsg: "Cannot delete a running command" }; - this.inputModel.flashInfoMsg(info, 2000); - return false; - } - } - GlobalCommandRunner.lineDelete(String(selectedLine), true); - return true; - } - - onWCmd(e: any, mods: KeyModsType) { - if (this.activeMainView.get() != "session") { - return; - } - let activeScreen = this.getActiveScreen(); - if (activeScreen == null) { - return; - } - let rtnp = this.showAlert({ - message: "Are you sure you want to delete this screen?", - confirm: true, - }); - rtnp.then((result) => { - if (!result) { - return; - } - GlobalCommandRunner.screenDelete(activeScreen.screenId, true); - }); - } - - onRCmd(e: any, mods: KeyModsType) { - if (this.activeMainView.get() != "session") { - return; - } - let activeScreen = this.getActiveScreen(); - if (activeScreen == null) { - return; - } - if (mods.shift) { - // restart last line - GlobalCommandRunner.lineRestart("E", true); - } else { - // restart selected line - let selectedLine = activeScreen.selectedLine.get(); - if (selectedLine == null || selectedLine == 0) { - return; - } - GlobalCommandRunner.lineRestart(String(selectedLine), true); - } - } - - onZoomChanged(): void { - mobx.action(() => { - this.devicePixelRatio.set(window.devicePixelRatio); - clearMonoFontCache(); - })(); - } - - getSelectedTermWrap(): TermWrap { - let screen = this.getActiveScreen(); - if (screen == null) { - return null; - } - let lineNum = screen.selectedLine.get(); - if (lineNum == null) { - return null; - } - let line = screen.getLineByNum(lineNum); - if (line == null) { - return null; - } - return screen.getTermWrap(line.lineid); - } - - clearModals(): boolean { - let didSomething = false; - mobx.action(() => { - if (GlobalModel.screenSettingsModal.get()) { - GlobalModel.screenSettingsModal.set(null); - didSomething = true; - } - if (GlobalModel.sessionSettingsModal.get()) { - GlobalModel.sessionSettingsModal.set(null); - didSomething = true; - } - if (GlobalModel.screenSettingsModal.get()) { - GlobalModel.screenSettingsModal.set(null); - didSomething = true; - } - if (GlobalModel.clientSettingsModal.get()) { - GlobalModel.clientSettingsModal.set(false); - didSomething = true; - } - if (GlobalModel.lineSettingsModal.get()) { - GlobalModel.lineSettingsModal.set(null); - didSomething = true; - } - })(); - return didSomething; - } - - restartWaveSrv(): void { - getApi().restartWaveSrv(); - } - - getLocalRemote(): RemoteType { - for (const remote of this.remotes) { - if (remote.local) { - return remote; - } - } - return null; - } - - getCurRemoteInstance(): RemoteInstanceType { - let screen = this.getActiveScreen(); - if (screen == null) { - return null; - } - return screen.getCurRemoteInstance(); - } - - onWaveSrvStatusChange(status: boolean): void { - mobx.action(() => { - this.waveSrvRunning.set(status); - })(); - } - - getLastLogs(numbOfLines: number, cb: (logs: any) => void): void { - getApi().getLastLogs(numbOfLines, cb); - } - - getContentHeight(context: RendererContext): number { - let key = context.screenId + "/" + context.lineId; - return this.termUsedRowsCache[key]; - } - - setContentHeight(context: RendererContext, height: number): void { - let key = context.screenId + "/" + context.lineId; - this.termUsedRowsCache[key] = height; - GlobalCommandRunner.setTermUsedRows(context, height); - } - - contextScreen(e: any, screenId: string) { - getApi().contextScreen({ screenId: screenId }, { x: e.x, y: e.y }); - } - - contextEditMenu(e: any, opts: ContextMenuOpts) { - getApi().contextEditMenu({ x: e.x, y: e.y }, opts); - } - - getUIContext(): UIContextType { - let rtn: UIContextType = { - sessionid: null, - screenid: null, - remote: null, - winsize: null, - linenum: null, - build: VERSION + " " + BUILD, - }; - let session = this.getActiveSession(); - if (session != null) { - rtn.sessionid = session.sessionId; - let screen = session.getActiveScreen(); - if (screen != null) { - rtn.screenid = screen.screenId; - rtn.remote = screen.curRemote.get(); - rtn.winsize = { rows: screen.lastRows, cols: screen.lastCols }; - rtn.linenum = screen.selectedLine.get(); - } - } - return rtn; - } - - onTCmd(e: any, mods: KeyModsType) { - GlobalCommandRunner.createNewScreen(); - } - - onICmd(e: any, mods: KeyModsType) { - this.inputModel.giveFocus(); - } - - onLCmd(e: any, mods: KeyModsType) { - let screen = this.getActiveScreen(); - if (screen != null) { - GlobalCommandRunner.screenSetFocus("cmd"); - } - } - - onHCmd(e: any, mods: KeyModsType) { - GlobalModel.historyViewModel.reSearch(); - } - - onPCmd(e: any, mods: KeyModsType) { - GlobalModel.modalsModel.pushModal(appconst.TAB_SWITCHER); - } - - getFocusedLine(): LineFocusType { - if (this.inputModel.hasFocus()) { - return { cmdInputFocus: true }; - } - let lineElem: any = document.activeElement.closest(".line[data-lineid]"); - if (lineElem == null) { - return { cmdInputFocus: false }; - } - let lineNum = parseInt(lineElem.dataset.linenum); - return { - cmdInputFocus: false, - lineid: lineElem.dataset.lineid, - linenum: isNaN(lineNum) ? null : lineNum, - screenid: lineElem.dataset.screenid, - }; - } - - cmdStatusUpdate(screenId: string, lineId: string, origStatus: string, newStatus: string) { - let wasRunning = cmdStatusIsRunning(origStatus); - let isRunning = cmdStatusIsRunning(newStatus); - if (wasRunning && !isRunning) { - let ptr = this.getActiveLine(screenId, lineId); - if (ptr != null) { - let screen = ptr.screen; - let renderer = screen.getRenderer(lineId); - if (renderer != null) { - renderer.setIsDone(); - } - let term = screen.getTermWrap(lineId); - if (term != null) { - term.cmdDone(); - } - } - } - } - - onMenuItemAbout(): void { - mobx.action(() => { - this.modalsModel.pushModal(appconst.ABOUT); - })(); - } - - onMetaPageUp(): void { - GlobalCommandRunner.screenSelectLine("-1"); - } - - onMetaPageDown(): void { - GlobalCommandRunner.screenSelectLine("+1"); - } - - onMetaArrowUp(): void { - GlobalCommandRunner.screenSelectLine("-1"); - } - - onMetaArrowDown(): void { - GlobalCommandRunner.screenSelectLine("+1"); - } - - onBracketCmd(e: any, arg: { relative: number }, mods: KeyModsType) { - if (arg.relative == 1) { - GlobalCommandRunner.switchScreen("+"); - } else if (arg.relative == -1) { - GlobalCommandRunner.switchScreen("-"); - } - } - - onDigitCmd(e: any, arg: { digit: number }, mods: KeyModsType) { - if (mods.meta && mods.ctrl) { - GlobalCommandRunner.switchSession(String(arg.digit)); - return; - } - GlobalCommandRunner.switchScreen(String(arg.digit)); - } - - isConnected(): boolean { - return this.ws.open.get(); - } - - runUpdate(genUpdate: UpdateMessage, interactive: boolean) { - mobx.action(() => { - let oldContext = this.getUIContext(); - try { - this.runUpdate_internal(genUpdate, oldContext, interactive); - } catch (e) { - console.log("error running update", e, genUpdate); - throw e; - } - let newContext = this.getUIContext(); - if (oldContext.sessionid != newContext.sessionid || oldContext.screenid != newContext.screenid) { - this.inputModel.resetInput(); - if ("cmdline" in genUpdate) { - // TODO a bit of a hack since this update gets applied in runUpdate_internal. - // we then undo that update with the resetInput, and then redo it with the line below - // not sure how else to handle this for now though - this.inputModel.updateCmdLine(genUpdate.cmdline); - } - } else if (remotePtrToString(oldContext.remote) != remotePtrToString(newContext.remote)) { - this.inputModel.resetHistory(); - } - })(); - } - - runUpdate_internal(genUpdate: UpdateMessage, uiContext: UIContextType, interactive: boolean) { - if ("ptydata64" in genUpdate) { - let ptyMsg: PtyDataUpdateType = genUpdate; - if (isBlank(ptyMsg.remoteid)) { - // regular update - this.updatePtyData(ptyMsg); - } else { - // remote update - let ptyData = base64ToArray(ptyMsg.ptydata64); - this.remotesModel.receiveData(ptyMsg.remoteid, ptyMsg.ptypos, ptyData); - } - return; - } - let update: ModelUpdateType = genUpdate; - if ("screens" in update) { - if (update.connect) { - this.screenMap.clear(); - } - let mods = genMergeDataMap( - this.screenMap, - update.screens, - (s: Screen) => s.screenId, - (sdata: ScreenDataType) => sdata.screenid, - (sdata: ScreenDataType) => new Screen(sdata) - ); - for (const screenId of mods.removed) { - this.removeScreenLinesByScreenId(screenId); - } - } - if ("sessions" in update || "activesessionid" in update) { - if (update.connect) { - this.sessionList.clear(); - } - let [oldActiveSessionId, oldActiveScreenId] = this.getActiveIds(); - genMergeData( - this.sessionList, - update.sessions, - (s: Session) => s.sessionId, - (sdata: SessionDataType) => sdata.sessionid, - (sdata: SessionDataType) => new Session(sdata), - (s: Session) => s.sessionIdx.get() - ); - if ("activesessionid" in update) { - let newSessionId = update.activesessionid; - if (this.activeSessionId.get() != newSessionId) { - this.activeSessionId.set(newSessionId); - } - } - let [newActiveSessionId, newActiveScreenId] = this.getActiveIds(); - if (oldActiveSessionId != newActiveSessionId || oldActiveScreenId != newActiveScreenId) { - this.activeMainView.set("session"); - this.deactivateScreenLines(); - this.ws.watchScreen(newActiveSessionId, newActiveScreenId); - } - } - if ("line" in update) { - this.addLineCmd(update.line, update.cmd, interactive); - } else if ("cmd" in update) { - this.updateCmd(update.cmd); - } - if ("lines" in update) { - for (const line of update.lines) { - this.addLineCmd(line, null, interactive); - } - } - if ("screenlines" in update) { - this.updateScreenLines(update.screenlines, false); - } - if ("remotes" in update) { - if (update.connect) { - this.remotes.clear(); - } - this.updateRemotes(update.remotes); - // This code's purpose is to show view remote connection modal when a new connection is added - if (update.remotes?.length && this.remotesModel.recentConnAddedState.get()) { - GlobalModel.remotesModel.openReadModal(update.remotes[0].remoteid); - } - } - if ("mainview" in update) { - if (update.mainview == "plugins") { - this.pluginsModel.showPluginsView(); - } else if (update.mainview == "bookmarks") { - this.bookmarksModel.showBookmarksView(update.bookmarks, update.selectedbookmark); - } else if (update.mainview == "session") { - this.activeMainView.set("session"); - } else if (update.mainview == "history") { - this.historyViewModel.showHistoryView(update.historyviewdata); - } else { - console.log("invalid mainview in update:", update.mainview); - } - } else if ("bookmarks" in update) { - this.bookmarksModel.mergeBookmarks(update.bookmarks); - } - if ("clientdata" in update) { - this.clientData.set(update.clientdata); - } - if (interactive && "info" in update) { - let info: InfoType = update.info; - this.inputModel.flashInfoMsg(info, info.timeoutms); - } - if (interactive && "remoteview" in update) { - let rview: RemoteViewType = update.remoteview; - if (rview.remoteedit != null) { - this.remotesModel.openEditModal({ ...rview.remoteedit }); - } - } - if (interactive && "alertmessage" in update) { - let alertMessage: AlertMessageType = update.alertmessage; - this.showAlert(alertMessage); - } - if ("cmdline" in update) { - this.inputModel.updateCmdLine(update.cmdline); - } - if (interactive && "history" in update) { - if (uiContext.sessionid == update.history.sessionid && uiContext.screenid == update.history.screenid) { - this.inputModel.setHistoryInfo(update.history); - } - } - if ("connect" in update) { - this.sessionListLoaded.set(true); - this.remotesLoaded.set(true); - } - if ("openaicmdinfochat" in update) { - this.inputModel.setOpenAICmdInfoChat(update.openaicmdinfochat); - } - if ("screenstatusindicators" in update) { - for (const indicator of update.screenstatusindicators) { - this.getScreenById_single(indicator.screenid)?.setStatusIndicator(indicator.status); - } - } - if ("screennumrunningcommands" in update) { - for (const snc of update.screennumrunningcommands) { - this.getScreenById_single(snc.screenid)?.setNumRunningCmds(snc.num); - } - } - if ("userinputrequest" in update) { - let userInputRequest: UserInputRequest = update.userinputrequest; - this.modalsModel.pushModal(appconst.USER_INPUT, userInputRequest); - } - } - - updateRemotes(remotes: RemoteType[]): void { - genMergeSimpleData(this.remotes, remotes, (r) => r.remoteid, null); - } - - getActiveSession(): Session { - return this.getSessionById(this.activeSessionId.get()); - } - - getSessionNames(): Record { - let rtn: Record = {}; - for (const session of this.sessionList) { - rtn[session.sessionId] = session.name.get(); - } - return rtn; - } - - getScreenNames(): Record { - let rtn: Record = {}; - for (let screen of this.screenMap.values()) { - rtn[screen.screenId] = screen.name.get(); - } - return rtn; - } - - getSessionById(sessionId: string): Session { - if (sessionId == null) { - return null; - } - for (const session of this.sessionList) { - if (session.sessionId == sessionId) { - return session; - } - } - return null; - } - - deactivateScreenLines() { - mobx.action(() => { - this.screenLines.clear(); - })(); - } - - getScreenLinesById(screenId: string): ScreenLines { - return this.screenLines.get(screenId); - } - - updateScreenLines(slines: ScreenLinesType, load: boolean) { - mobx.action(() => { - let existingWin = this.screenLines.get(slines.screenid); - if (existingWin == null) { - if (!load) { - console.log("cannot update screen-lines that does not exist", slines.screenid); - return; - } - let newWindow = new ScreenLines(slines.screenid); - this.screenLines.set(slines.screenid, newWindow); - newWindow.updateData(slines, load); - } else { - existingWin.updateData(slines, load); - existingWin.loaded.set(true); - } - })(); - } - - removeScreenLinesByScreenId(screenId: string) { - mobx.action(() => { - this.screenLines.delete(screenId); - })(); - } - - getScreenById(sessionId: string, screenId: string): Screen { - return this.screenMap.get(screenId); - } - - getScreenById_single(screenId: string): Screen { - return this.screenMap.get(screenId); - } - - getSessionScreens(sessionId: string): Screen[] { - let rtn: Screen[] = []; - for (let screen of this.screenMap.values()) { - if (screen.sessionId == sessionId) { - rtn.push(screen); - } - } - return rtn; - } - - getScreenLinesForActiveScreen(): ScreenLines { - let screen = this.getActiveScreen(); - if (screen == null) { - return null; - } - return this.screenLines.get(screen.screenId); - } - - getActiveScreen(): Screen { - let session = this.getActiveSession(); - if (session == null) { - return null; - } - return session.getActiveScreen(); - } - - handleCmdRestart(cmd: CmdDataType) { - if (cmd == null || !cmd.restarted) { - return; - } - let screen = this.screenMap.get(cmd.screenid); - if (screen == null) { - return; - } - let termWrap = screen.getTermWrap(cmd.lineid); - if (termWrap == null) { - return; - } - termWrap.reload(0); - } - - addLineCmd(line: LineType, cmd: CmdDataType, interactive: boolean) { - let slines = this.getScreenLinesById(line.screenid); - if (slines == null) { - return; - } - slines.addLineCmd(line, cmd, interactive); - this.handleCmdRestart(cmd); - } - - updateCmd(cmd: CmdDataType) { - let slines = this.screenLines.get(cmd.screenid); - if (slines != null) { - slines.updateCmd(cmd); - } - this.handleCmdRestart(cmd); - } - - isInfoUpdate(update: UpdateMessage): boolean { - if (update == null || "ptydata64" in update) { - return false; - } - return update.info != null || update.history != null; - } - - getClientDataLoop(loopNum: number): void { - this.getClientData(); - let clientStop = this.getHasClientStop(); - if (this.clientData.get() != null && !clientStop) { - return; - } - let timeoutMs = 1000; - if (!clientStop && loopNum > 5) { - timeoutMs = 3000; - } - if (!clientStop && loopNum > 10) { - timeoutMs = 10000; - } - if (!clientStop && loopNum > 15) { - timeoutMs = 30000; - } - setTimeout(() => this.getClientDataLoop(loopNum + 1), timeoutMs); - } - - getClientData(): void { - let url = new URL(GlobalModel.getBaseHostPort() + "/api/get-client-data"); - let fetchHeaders = this.getFetchHeaders(); - fetch(url, { method: "post", body: null, headers: fetchHeaders }) - .then((resp) => handleJsonFetchResponse(url, resp)) - .then((data) => { - mobx.action(() => { - let clientData: ClientDataType = data.data; - this.clientData.set(clientData); - })(); - }) - .catch((err) => { - this.errorHandler("calling get-client-data", err, true); - }); - } - - submitCommandPacket(cmdPk: FeCmdPacketType, interactive: boolean): Promise { - if (this.debugCmds > 0) { - console.log("[cmd]", cmdPacketString(cmdPk)); - if (this.debugCmds > 1) { - console.trace(); - } - } - // adding cmdStr for debugging only (easily filter run-command calls in the network tab of debugger) - let cmdStr = cmdPk.metacmd + (cmdPk.metasubcmd ? ":" + cmdPk.metasubcmd : ""); - let url = new URL(GlobalModel.getBaseHostPort() + "/api/run-command?cmd=" + cmdStr); - let fetchHeaders = this.getFetchHeaders(); - let prtn = fetch(url, { - method: "post", - body: JSON.stringify(cmdPk), - headers: fetchHeaders, - }) - .then((resp) => handleJsonFetchResponse(url, resp)) - .then((data) => { - mobx.action(() => { - let update = data.data; - if (update != null) { - this.runUpdate(update, interactive); - } - if (interactive && !this.isInfoUpdate(update)) { - GlobalModel.inputModel.clearInfoMsg(true); - } - })(); - return { success: true }; - }) - .catch((err) => { - this.errorHandler("calling run-command", err, interactive); - let errMessage = "error running command"; - if (err != null && !isBlank(err.message)) { - errMessage = err.message; - } - return { success: false, error: errMessage }; - }); - return prtn; - } - - submitCommand( - metaCmd: string, - metaSubCmd: string, - args: string[], - kwargs: Record, - interactive: boolean - ): Promise { - let pk: FeCmdPacketType = { - type: "fecmd", - metacmd: metaCmd, - metasubcmd: metaSubCmd, - args: args, - kwargs: { ...kwargs }, - uicontext: this.getUIContext(), - interactive: interactive, - }; - /** - console.log( - "CMD", - pk.metacmd + (pk.metasubcmd != null ? ":" + pk.metasubcmd : ""), - pk.args, - pk.kwargs, - pk.interactive - ); - */ - return this.submitCommandPacket(pk, interactive); - } - - submitChatInfoCommand(chatMsg: string, curLineStr: string, clear: boolean): Promise { - let commandStr = "/chat " + chatMsg; - let interactive = false; - let pk: FeCmdPacketType = { - type: "fecmd", - metacmd: "eval", - args: [commandStr], - kwargs: {}, - uicontext: this.getUIContext(), - interactive: interactive, - rawstr: chatMsg, - }; - pk.kwargs["nohist"] = "1"; - if (clear) { - pk.kwargs["cmdinfoclear"] = "1"; - } else { - pk.kwargs["cmdinfo"] = "1"; - } - pk.kwargs["curline"] = curLineStr; - return this.submitCommandPacket(pk, interactive); - } - - submitRawCommand(cmdStr: string, addToHistory: boolean, interactive: boolean): Promise { - let pk: FeCmdPacketType = { - type: "fecmd", - metacmd: "eval", - args: [cmdStr], - kwargs: null, - uicontext: this.getUIContext(), - interactive: interactive, - rawstr: cmdStr, - }; - if (!addToHistory && pk.kwargs) { - pk.kwargs["nohist"] = "1"; - } - return this.submitCommandPacket(pk, interactive); - } - - // returns [sessionId, screenId] - getActiveIds(): [string, string] { - let activeSession = this.getActiveSession(); - let activeScreen = this.getActiveScreen(); - return [activeSession?.sessionId, activeScreen?.screenId]; - } - - _loadScreenLinesAsync(newWin: ScreenLines) { - this.screenLines.set(newWin.screenId, newWin); - let usp = new URLSearchParams({ screenid: newWin.screenId }); - let url = new URL(GlobalModel.getBaseHostPort() + "/api/get-screen-lines?" + usp.toString()); - let fetchHeaders = GlobalModel.getFetchHeaders(); - fetch(url, { headers: fetchHeaders }) - .then((resp) => handleJsonFetchResponse(url, resp)) - .then((data) => { - if (data.data == null) { - console.log("null screen-lines returned from get-screen-lines"); - return; - } - let slines: ScreenLinesType = data.data; - this.updateScreenLines(slines, true); - }) - .catch((err) => { - this.errorHandler(sprintf("getting screen-lines=%s", newWin.screenId), err, false); - }); - } - - loadScreenLines(screenId: string): ScreenLines { - let newWin = new ScreenLines(screenId); - setTimeout(() => this._loadScreenLinesAsync(newWin), 0); - return newWin; - } - - getRemote(remoteId: string): RemoteType { - if (remoteId == null) { - return null; - } - return this.remotes.find((remote) => remote.remoteid === remoteId); - } - - getRemoteNames(): Record { - let rtn: Record = {}; - for (const remote of this.remotes) { - if (!isBlank(remote.remotealias)) { - rtn[remote.remoteid] = remote.remotealias; - } else { - rtn[remote.remoteid] = remote.remotecanonicalname; - } - } - return rtn; - } - - getRemoteByName(name: string): RemoteType { - for (const remote of this.remotes) { - if (remote.remotecanonicalname == name || remote.remotealias == name) { - return remote; - } - } - return null; - } - - getCmd(line: LineType): Cmd { - return this.getCmdByScreenLine(line.screenid, line.lineid); - } - - getCmdByScreenLine(screenId: string, lineId: string): Cmd { - let slines = this.getScreenLinesById(screenId); - if (slines == null) { - return null; - } - return slines.getCmd(lineId); - } - - getActiveLine(screenId: string, lineid: string): SWLinePtr { - let slines = this.screenLines.get(screenId); - if (slines == null) { - return null; - } - if (!slines.loaded.get()) { - return null; - } - let cmd = slines.getCmd(lineid); - if (cmd == null) { - return null; - } - let line: LineType = null; - for (const element of slines.lines) { - if (element.lineid == lineid) { - line = element; - break; - } - } - if (line == null) { - return null; - } - let screen = this.getScreenById_single(slines.screenId); - return { line: line, slines: slines, screen: screen }; - } - - updatePtyData(ptyMsg: PtyDataUpdateType): void { - let linePtr = this.getActiveLine(ptyMsg.screenid, ptyMsg.lineid); - if (linePtr != null) { - linePtr.screen.updatePtyData(ptyMsg); - } - } - - errorHandler(str: string, err: any, interactive: boolean) { - console.log("[error]", str, err); - if (interactive) { - let errMsg = "error running command"; - if (err?.message) { - errMsg = err.message; - } - this.inputModel.flashInfoMsg({ infoerror: errMsg }, null); - } - } - - sendInputPacket(inputPacket: any) { - this.ws.pushMessage(inputPacket); - } - - sendUserInput(userInputResponsePacket: UserInputResponsePacket) { - this.ws.pushMessage(userInputResponsePacket); - } - - sendCmdInputText(screenId: string, sp: T.StrWithPos) { - let pk: T.CmdInputTextPacketType = { - type: "cmdinputtext", - seqnum: this.getNextPacketSeqNum(), - screenid: screenId, - text: sp, - }; - this.ws.pushMessage(pk); - } - - resolveUserIdToName(userid: string): string { - return "@[unknown]"; - } - - resolveRemoteIdToRef(remoteId: string) { - let remote = this.getRemote(remoteId); - if (remote == null) { - return "[unknown]"; - } - if (!isBlank(remote.remotealias)) { - return remote.remotealias; - } - return remote.remotecanonicalname; - } - - resolveRemoteIdToFullRef(remoteId: string) { - let remote = this.getRemote(remoteId); - if (remote == null) { - return "[unknown]"; - } - if (!isBlank(remote.remotealias)) { - return remote.remotealias + " (" + remote.remotecanonicalname + ")"; - } - return remote.remotecanonicalname; - } - - readRemoteFile(screenId: string, lineId: string, path: string): Promise { - let urlParams = { - screenid: screenId, - lineid: lineId, - path: path, - }; - let usp = new URLSearchParams(urlParams); - let url = new URL(GlobalModel.getBaseHostPort() + "/api/read-file?" + usp.toString()); - let fetchHeaders = this.getFetchHeaders(); - let fileInfo: T.FileInfoType = null; - let badResponseStr: string = null; - let prtn = fetch(url, { method: "get", headers: fetchHeaders }) - .then((resp) => { - if (!resp.ok) { - badResponseStr = sprintf( - "Bad fetch response for /api/read-file: %d %s", - resp.status, - resp.statusText - ); - return resp.text() as any; - } - fileInfo = JSON.parse(base64ToString(resp.headers.get("X-FileInfo"))); - return resp.blob(); - }) - .then((blobOrText: any) => { - if (blobOrText instanceof Blob) { - let blob: Blob = blobOrText; - let file = new File([blob], fileInfo.name, { type: blob.type, lastModified: fileInfo.modts }); - let isWriteable = (fileInfo.perm & 0o222) > 0; // checks for unix permission "w" bits - (file as any).readOnly = !isWriteable; - (file as any).notFound = !!fileInfo.notfound; - return file as T.ExtFile; - } else { - let textError: string = blobOrText; - if (textError == null || textError.length == 0) { - throw new Error(badResponseStr); - } - throw new Error(textError); - } - }); - return prtn; - } - - writeRemoteFile( - screenId: string, - lineId: string, - path: string, - data: Uint8Array, - opts?: { useTemp?: boolean } - ): Promise { - opts = opts || {}; - let params = { - screenid: screenId, - lineid: lineId, - path: path, - usetemp: !!opts.useTemp, - }; - let formData = new FormData(); - formData.append("params", JSON.stringify(params)); - let blob = new Blob([data], { type: "application/octet-stream" }); - formData.append("data", blob); - let url = new URL(GlobalModel.getBaseHostPort() + "/api/write-file"); - let fetchHeaders = this.getFetchHeaders(); - let prtn = fetch(url, { method: "post", headers: fetchHeaders, body: formData }); - return prtn - .then((resp) => handleJsonFetchResponse(url, resp)) - .then((_) => { - return; - }); - } -} - -class CommandRunner { - loadHistory(show: boolean, htype: string) { - let kwargs = { nohist: "1" }; - if (!show) { - kwargs["noshow"] = "1"; - } - if (htype != null && htype != "screen") { - kwargs["type"] = htype; - } - GlobalModel.submitCommand("history", null, null, kwargs, true); - } - - resetShellState() { - GlobalModel.submitCommand("reset", null, null, null, true); - } - - historyPurgeLines(lines: string[]): Promise { - let prtn = GlobalModel.submitCommand("history", "purge", lines, { nohist: "1" }, false); - return prtn; - } - - switchSession(session: string) { - mobx.action(() => { - GlobalModel.activeMainView.set("session"); - })(); - GlobalModel.submitCommand("session", null, [session], { nohist: "1" }, false); - } - - switchScreen(screen: string, session?: string) { - mobx.action(() => { - GlobalModel.activeMainView.set("session"); - })(); - let kwargs = { nohist: "1" }; - if (session != null) { - kwargs["session"] = session; - } - GlobalModel.submitCommand("screen", null, [screen], kwargs, false); - } - - lineView(sessionId: string, screenId: string, lineNum?: number) { - let screen = GlobalModel.getScreenById(sessionId, screenId); - if (screen != null && lineNum != null) { - screen.setAnchorFields(lineNum, 0, "line:view"); - } - let lineNumStr = lineNum == null || lineNum == 0 ? "E" : String(lineNum); - GlobalModel.submitCommand("line", "view", [sessionId, screenId, lineNumStr], { nohist: "1" }, false); - } - - lineArchive(lineArg: string, archive: boolean): Promise { - let kwargs = { nohist: "1" }; - let archiveStr = archive ? "1" : "0"; - return GlobalModel.submitCommand("line", "archive", [lineArg, archiveStr], kwargs, false); - } - - lineDelete(lineArg: string, interactive: boolean): Promise { - return GlobalModel.submitCommand("line", "delete", [lineArg], { nohist: "1" }, interactive); - } - - lineRestart(lineArg: string, interactive: boolean): Promise { - return GlobalModel.submitCommand("line", "restart", [lineArg], { nohist: "1" }, interactive); - } - - lineSet(lineArg: string, opts: { renderer?: string }): Promise { - let kwargs = { nohist: "1" }; - if ("renderer" in opts) { - kwargs["renderer"] = opts.renderer ?? ""; - } - return GlobalModel.submitCommand("line", "set", [lineArg], kwargs, false); - } - - createNewSession() { - GlobalModel.submitCommand("session", "open", null, { nohist: "1" }, false); - } - - createNewScreen() { - GlobalModel.submitCommand("screen", "open", null, { nohist: "1" }, false); - } - - closeScreen(screen: string) { - GlobalModel.submitCommand("screen", "close", [screen], { nohist: "1" }, false); - } - - // include is lineIds to include, exclude is lineIds to exclude - // if include is given then it *only* does those ids. if exclude is given (or not), - // it does all running commands in the screen except for excluded. - resizeScreen(screenId: string, rows: number, cols: number, opts?: { include?: string[]; exclude?: string[] }) { - let kwargs: Record = { - nohist: "1", - screen: screenId, - cols: String(cols), - rows: String(rows), - }; - if (opts?.include != null && opts?.include.length > 0) { - kwargs.include = opts.include.join(","); - } - if (opts?.exclude != null && opts?.exclude.length > 0) { - kwargs.exclude = opts.exclude.join(","); - } - GlobalModel.submitCommand("screen", "resize", null, kwargs, false); - } - - screenArchive(screenId: string, shouldArchive: boolean): Promise { - return GlobalModel.submitCommand( - "screen", - "archive", - [screenId, shouldArchive ? "1" : "0"], - { nohist: "1" }, - false - ); - } - - screenDelete(screenId: string, interactive: boolean): Promise { - return GlobalModel.submitCommand("screen", "delete", [screenId], { nohist: "1" }, interactive); - } - - screenWebShare(screenId: string, shouldShare: boolean): Promise { - let kwargs: Record = { nohist: "1" }; - kwargs["screen"] = screenId; - return GlobalModel.submitCommand("screen", "webshare", [shouldShare ? "1" : "0"], kwargs, false); - } - - showRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "show", null, { nohist: "1", remote: remoteid }, true); - } - - showAllRemotes() { - GlobalModel.submitCommand("remote", "showall", null, { nohist: "1" }, true); - } - - connectRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "connect", null, { nohist: "1", remote: remoteid }, true); - } - - disconnectRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "disconnect", null, { nohist: "1", remote: remoteid }, true); - } - - installRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "install", null, { nohist: "1", remote: remoteid }, true); - } - - installCancelRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "installcancel", null, { nohist: "1", remote: remoteid }, true); - } - - createRemote(cname: string, kwargsArg: Record, interactive: boolean): Promise { - let kwargs = Object.assign({}, kwargsArg); - kwargs["nohist"] = "1"; - return GlobalModel.submitCommand("remote", "new", [cname], kwargs, interactive); - } - - openCreateRemote(): void { - GlobalModel.submitCommand("remote", "new", null, { nohist: "1", visual: "1" }, true); - } - - screenSetRemote(remoteArg: string, nohist: boolean, interactive: boolean): Promise { - let kwargs = {}; - if (nohist) { - kwargs["nohist"] = "1"; - } - return GlobalModel.submitCommand("connect", null, [remoteArg], kwargs, interactive); - } - - editRemote(remoteid: string, kwargsArg: Record): void { - let kwargs = Object.assign({}, kwargsArg); - kwargs["nohist"] = "1"; - kwargs["remote"] = remoteid; - GlobalModel.submitCommand("remote", "set", null, kwargs, true); - } - - openEditRemote(remoteid: string): void { - GlobalModel.submitCommand("remote", "set", null, { remote: remoteid, nohist: "1", visual: "1" }, true); - } - - archiveRemote(remoteid: string) { - GlobalModel.submitCommand("remote", "archive", null, { remote: remoteid, nohist: "1" }, true); - } - - importSshConfig() { - GlobalModel.submitCommand("remote", "parse", null, { nohist: "1", visual: "1" }, true); - } - - screenSelectLine(lineArg: string, focusVal?: string) { - let kwargs: Record = { - nohist: "1", - line: lineArg, - }; - if (focusVal != null) { - kwargs["focus"] = focusVal; - } - GlobalModel.submitCommand("screen", "set", null, kwargs, false); - } - - screenReorder(screenId: string, index: string) { - let kwargs: Record = { - nohist: "1", - screenId: screenId, - index: index, - }; - GlobalModel.submitCommand("screen", "reorder", null, kwargs, false); - } - - setTermUsedRows(termContext: RendererContext, height: number) { - let kwargs: Record = {}; - kwargs["screen"] = termContext.screenId; - kwargs["hohist"] = "1"; - let posargs = [String(termContext.lineNum), String(height)]; - GlobalModel.submitCommand("line", "setheight", posargs, kwargs, false); - } - - screenSetAnchor(sessionId: string, screenId: string, anchorVal: string): void { - let kwargs = { - nohist: "1", - anchor: anchorVal, - session: sessionId, - screen: screenId, - }; - GlobalModel.submitCommand("screen", "set", null, kwargs, false); - } - - screenSetFocus(focusVal: string): void { - GlobalModel.submitCommand("screen", "set", null, { focus: focusVal, nohist: "1" }, false); - } - - screenSetSettings( - screenId: string, - settings: { tabcolor?: string; tabicon?: string; name?: string; sharename?: string }, - interactive: boolean - ): Promise { - let kwargs: { [key: string]: any } = Object.assign({}, settings); - kwargs["nohist"] = "1"; - kwargs["screen"] = screenId; - return GlobalModel.submitCommand("screen", "set", null, kwargs, interactive); - } - - sessionArchive(sessionId: string, shouldArchive: boolean): Promise { - return GlobalModel.submitCommand( - "session", - "archive", - [sessionId, shouldArchive ? "1" : "0"], - { nohist: "1" }, - false - ); - } - - sessionDelete(sessionId: string): Promise { - return GlobalModel.submitCommand("session", "delete", [sessionId], { nohist: "1" }, false); - } - - sessionSetSettings(sessionId: string, settings: { name?: string }, interactive: boolean): Promise { - let kwargs = Object.assign({}, settings); - kwargs["nohist"] = "1"; - kwargs["session"] = sessionId; - return GlobalModel.submitCommand("session", "set", null, kwargs, interactive); - } - - lineStar(lineId: string, starVal: number) { - GlobalModel.submitCommand("line", "star", [lineId, String(starVal)], { nohist: "1" }, true); - } - - lineBookmark(lineId: string) { - GlobalModel.submitCommand("line", "bookmark", [lineId], { nohist: "1" }, true); - } - - linePin(lineId: string, val: boolean) { - GlobalModel.submitCommand("line", "pin", [lineId, val ? "1" : "0"], { nohist: "1" }, true); - } - - bookmarksView() { - GlobalModel.submitCommand("bookmarks", "show", null, { nohist: "1" }, true); - } - - connectionsView() { - GlobalModel.connectionViewModel.showConnectionsView(); - } - - clientSettingsView() { - GlobalModel.clientSettingsViewModel.showClientSettingsView(); - } - - historyView(params: HistorySearchParams) { - let kwargs = { nohist: "1" }; - kwargs["offset"] = String(params.offset); - kwargs["rawoffset"] = String(params.rawOffset); - if (params.searchText != null) { - kwargs["text"] = params.searchText; - } - if (params.searchSessionId != null) { - kwargs["searchsession"] = params.searchSessionId; - } - if (params.searchRemoteId != null) { - kwargs["searchremote"] = params.searchRemoteId; - } - if (params.fromTs != null) { - kwargs["fromts"] = String(params.fromTs); - } - if (params.noMeta) { - kwargs["meta"] = "0"; - } - if (params.filterCmds) { - kwargs["filter"] = "1"; - } - GlobalModel.submitCommand("history", "viewall", null, kwargs, true); - } - - telemetryOff(interactive: boolean): Promise { - return GlobalModel.submitCommand("telemetry", "off", null, { nohist: "1" }, interactive); - } - - telemetryOn(interactive: boolean): Promise { - return GlobalModel.submitCommand("telemetry", "on", null, { nohist: "1" }, interactive); - } - - releaseCheckAutoOff(interactive: boolean): Promise { - return GlobalModel.submitCommand("releasecheck", "autooff", null, { nohist: "1" }, interactive); - } - - releaseCheckAutoOn(interactive: boolean): Promise { - return GlobalModel.submitCommand("releasecheck", "autoon", null, { nohist: "1" }, interactive); - } - - setTermFontSize(fsize: number, interactive: boolean): Promise { - let kwargs = { - nohist: "1", - termfontsize: String(fsize), - }; - return GlobalModel.submitCommand("client", "set", null, kwargs, interactive); - } - - setClientOpenAISettings(opts: { model?: string; apitoken?: string; maxtokens?: string }): Promise { - let kwargs = { - nohist: "1", - }; - if (opts.model != null) { - kwargs["openaimodel"] = opts.model; - } - if (opts.apitoken != null) { - kwargs["openaiapitoken"] = opts.apitoken; - } - if (opts.maxtokens != null) { - kwargs["openaimaxtokens"] = opts.maxtokens; - } - return GlobalModel.submitCommand("client", "set", null, kwargs, false); - } - - clientAcceptTos(): void { - GlobalModel.submitCommand("client", "accepttos", null, { nohist: "1" }, true); - } - - clientSetConfirmFlag(flag: string, value: boolean): Promise { - let kwargs = { nohist: "1" }; - let valueStr = value ? "1" : "0"; - return GlobalModel.submitCommand("client", "setconfirmflag", [flag, valueStr], kwargs, false); - } - - clientSetSidebar(width: number, collapsed: boolean): Promise { - let kwargs = { nohist: "1", width: `${width}`, collapsed: collapsed ? "1" : "0" }; - return GlobalModel.submitCommand("client", "setsidebar", null, kwargs, false); - } - - editBookmark(bookmarkId: string, desc: string, cmdstr: string) { - let kwargs = { - nohist: "1", - desc: desc, - cmdstr: cmdstr, - }; - GlobalModel.submitCommand("bookmark", "set", [bookmarkId], kwargs, true); - } - - deleteBookmark(bookmarkId: string): void { - GlobalModel.submitCommand("bookmark", "delete", [bookmarkId], { nohist: "1" }, true); - } - - openSharedSession(): void { - GlobalModel.submitCommand("session", "openshared", null, { nohist: "1" }, true); - } - - setLineState( - screenId: string, - lineId: string, - state: T.LineStateType, - interactive: boolean - ): Promise { - let stateStr = JSON.stringify(state); - return GlobalModel.submitCommand( - "line", - "set", - [lineId], - { screen: screenId, nohist: "1", state: stateStr }, - interactive - ); - } - - screenSidebarAddLine(lineId: string) { - GlobalModel.submitCommand("sidebar", "add", null, { nohist: "1", line: lineId }, false); - } - - screenSidebarRemove() { - GlobalModel.submitCommand("sidebar", "remove", null, { nohist: "1" }, false); - } - - screenSidebarClose(): void { - GlobalModel.submitCommand("sidebar", "close", null, { nohist: "1" }, false); - } - - screenSidebarOpen(width?: string): void { - let kwargs: Record = { nohist: "1" }; - if (width != null) { - kwargs.width = width; - } - GlobalModel.submitCommand("sidebar", "open", null, kwargs, false); - } -} - -function cmdPacketString(pk: FeCmdPacketType): string { - let cmd = pk.metacmd; - if (pk.metasubcmd != null) { - cmd += ":" + pk.metasubcmd; - } - let parts = [cmd]; - if (pk.kwargs != null) { - for (let key in pk.kwargs) { - parts.push(sprintf("%s=%s", key, pk.kwargs[key])); - } - } - if (pk.args != null) { - parts.push(...pk.args); - } - return parts.join(" "); -} - -function _getPtyDataFromUrl(url: string): Promise { - let ptyOffset = 0; - let fetchHeaders = GlobalModel.getFetchHeaders(); - return fetch(url, { headers: fetchHeaders }) - .then((resp) => { - if (!resp.ok) { - throw new Error(sprintf("Bad fetch response for /api/ptyout: %d %s", resp.status, resp.statusText)); - } - let ptyOffsetStr = resp.headers.get("X-PtyDataOffset"); - if (ptyOffsetStr != null && !isNaN(parseInt(ptyOffsetStr))) { - ptyOffset = parseInt(ptyOffsetStr); - } - return resp.arrayBuffer(); - }) - .then((buf) => { - return { pos: ptyOffset, data: new Uint8Array(buf) }; - }); -} - -function getTermPtyData(termContext: TermContextUnion): Promise { - if ("remoteId" in termContext) { - return getRemotePtyData(termContext.remoteId); - } - return getPtyData(termContext.screenId, termContext.lineId, termContext.lineNum); -} - -function getPtyData(screenId: string, lineId: string, lineNum: number): Promise { - let url = sprintf( - GlobalModel.getBaseHostPort() + "/api/ptyout?linenum=%d&screenid=%s&lineid=%s", - lineNum, - screenId, - lineId - ); - return _getPtyDataFromUrl(url); -} - -function getRemotePtyData(remoteId: string): Promise { - let url = sprintf(GlobalModel.getBaseHostPort() + "/api/remote-pty?remoteid=%s", remoteId); - return _getPtyDataFromUrl(url); -} - -let GlobalModel: Model = null; -let GlobalCommandRunner: CommandRunner = null; -if ((window as any).GlobalModel == null) { - (window as any).GlobalModel = new Model(); - (window as any).GlobalCommandRunner = new CommandRunner(); - (window as any).getMonoFontSize = getMonoFontSize; -} -GlobalModel = (window as any).GlobalModel; -GlobalCommandRunner = (window as any).GlobalCommandRunner; - -export { - Model, - Session, - ScreenLines, - GlobalModel, - GlobalCommandRunner, - Cmd, - Screen, - riToRPtr, - TabColors, - TabIcons, - RemoteColors, - getTermPtyData, - RemotesModalModel, - RemotesModel, - MinFontSize, - MaxFontSize, - SpecialLineContainer, - ForwardLineContainer, - VERSION, -}; -export type { LineContainerModel }; From 96cabeff982a9793cfa0a57c53a3fcb06aef8176 Mon Sep 17 00:00:00 2001 From: Evan Simkowitz Date: Tue, 20 Feb 2024 12:23:39 -0800 Subject: [PATCH 4/4] Fix sidebar middle div max-height calculation (#299) * Fix sidebar middle div max-height calculation * remove logs * update bottom div padding --- src/app/sidebar/sidebar.less | 4 ++-- src/app/sidebar/sidebar.tsx | 43 +++++++++++++++++++++++++++++++----- 2 files changed, 39 insertions(+), 8 deletions(-) diff --git a/src/app/sidebar/sidebar.less b/src/app/sidebar/sidebar.less index 96e4e1803..d548bc447 100644 --- a/src/app/sidebar/sidebar.less +++ b/src/app/sidebar/sidebar.less @@ -119,7 +119,6 @@ } .middle { - max-height: calc(100vh - 32em); padding: 4px 6px 8px 6px; border-bottom: 1px solid @base-border; .item { @@ -142,9 +141,10 @@ .bottom { position: absolute; - bottom: 2em; + bottom: 2rem; left: 0; width: 100%; + padding-top: 0.8rem; } .item { diff --git a/src/app/sidebar/sidebar.tsx b/src/app/sidebar/sidebar.tsx index fe0f67438..a518f3301 100644 --- a/src/app/sidebar/sidebar.tsx +++ b/src/app/sidebar/sidebar.tsx @@ -64,7 +64,7 @@ interface MainSideBarProps { @mobxReact.observer class MainSideBar extends React.Component { - sidebarRef = React.createRef(); + middleHeightSubtractor = mobx.observable.box(404); handleSessionClick(sessionId: string) { GlobalCommandRunner.switchSession(sessionId); @@ -203,14 +203,37 @@ class MainSideBar extends React.Component { }); } + /** + * Calculate the subtractor portion for the middle div's height calculation, which should be `100vh - subtractor`. + */ + setMiddleHeightSubtractor() { + const windowHeight = window.innerHeight; + const bottomHeight = windowHeight - window.document.getElementById("sidebar-bottom")?.offsetTop; + const middleTop = document.getElementById("sidebar-middle")?.offsetTop; + const newMiddleHeightSubtractor = bottomHeight + middleTop; + if (!Number.isNaN(newMiddleHeightSubtractor)) { + mobx.action(() => { + this.middleHeightSubtractor.set(newMiddleHeightSubtractor); + })(); + } + } + + componentDidMount() { + this.setMiddleHeightSubtractor(); + } + + componentDidUpdate() { + this.setMiddleHeightSubtractor(); + } + render() { - let clientData = this.props.clientData; + const clientData = this.props.clientData; let needsUpdate = false; if (!clientData?.clientopts.noreleasecheck && !isBlank(clientData?.releaseinfo?.latestversion)) { needsUpdate = compareLoose(appconst.VERSION, clientData.releaseinfo.latestversion) < 0; } - let mainSidebar = GlobalModel.mainSidebarModel; - let isCollapsed = mainSidebar.getCollapsed(); + const mainSidebar = GlobalModel.mainSidebarModel; + const isCollapsed = mainSidebar.getCollapsed(); return ( { , ]} /> -
{this.getSessions()}
-
+ +