mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-02 18:39:05 +01:00
560 lines
17 KiB
TypeScript
560 lines
17 KiB
TypeScript
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import {
|
|
getLayoutModelForActiveTab,
|
|
getLayoutModelForTabById,
|
|
LayoutTreeActionType,
|
|
LayoutTreeInsertNodeAction,
|
|
newLayoutNode,
|
|
} from "@/layout/index";
|
|
import { getWebServerEndpoint } from "@/util/endpoints";
|
|
import { fetch } from "@/util/fetchutil";
|
|
import { getPrefixedSettings, isBlank } from "@/util/util";
|
|
import { atom, Atom, createStore, PrimitiveAtom, useAtomValue } from "jotai";
|
|
import { modalsModel } from "./modalmodel";
|
|
import { ClientService, ObjectService } from "./services";
|
|
import * as WOS from "./wos";
|
|
import { getFileSubject, waveEventSubscribe } from "./wps";
|
|
|
|
let PLATFORM: NodeJS.Platform = "darwin";
|
|
const globalStore = createStore();
|
|
let atoms: GlobalAtomsType;
|
|
let globalEnvironment: "electron" | "renderer";
|
|
const blockComponentModelMap = new Map<string, BlockComponentModel>();
|
|
const Counters = new Map<string, number>();
|
|
const ConnStatusMap = new Map<string, PrimitiveAtom<ConnStatus>>();
|
|
|
|
type GlobalInitOptions = {
|
|
platform: NodeJS.Platform;
|
|
windowId: string;
|
|
clientId: string;
|
|
environment: "electron" | "renderer";
|
|
};
|
|
|
|
function initGlobal(initOpts: GlobalInitOptions) {
|
|
globalEnvironment = initOpts.environment;
|
|
setPlatform(initOpts.platform);
|
|
initGlobalAtoms(initOpts);
|
|
}
|
|
|
|
function setPlatform(platform: NodeJS.Platform) {
|
|
PLATFORM = platform;
|
|
}
|
|
|
|
function initGlobalAtoms(initOpts: GlobalInitOptions) {
|
|
const windowIdAtom = atom(initOpts.windowId) as PrimitiveAtom<string>;
|
|
const clientIdAtom = atom(initOpts.clientId) as PrimitiveAtom<string>;
|
|
const uiContextAtom = atom((get) => {
|
|
const windowData = get(windowDataAtom);
|
|
const uiContext: UIContext = {
|
|
windowid: get(atoms.windowId),
|
|
activetabid: windowData?.activetabid,
|
|
};
|
|
return uiContext;
|
|
}) as Atom<UIContext>;
|
|
|
|
const isFullScreenAtom = atom(false) as PrimitiveAtom<boolean>;
|
|
try {
|
|
getApi().onFullScreenChange((isFullScreen) => {
|
|
globalStore.set(isFullScreenAtom, isFullScreen);
|
|
});
|
|
} catch (_) {
|
|
// do nothing
|
|
}
|
|
|
|
const showAboutModalAtom = atom(false) as PrimitiveAtom<boolean>;
|
|
try {
|
|
getApi().onMenuItemAbout(() => {
|
|
modalsModel.pushModal("AboutModal");
|
|
});
|
|
} catch (_) {
|
|
// do nothing
|
|
}
|
|
|
|
const clientAtom: Atom<Client> = atom((get) => {
|
|
const clientId = get(clientIdAtom);
|
|
if (clientId == null) {
|
|
return null;
|
|
}
|
|
return WOS.getObjectValue(WOS.makeORef("client", clientId), get);
|
|
});
|
|
const windowDataAtom: Atom<WaveWindow> = atom((get) => {
|
|
const windowId = get(windowIdAtom);
|
|
if (windowId == null) {
|
|
return null;
|
|
}
|
|
const rtn = WOS.getObjectValue<WaveWindow>(WOS.makeORef("window", windowId), get);
|
|
return rtn;
|
|
});
|
|
const workspaceAtom: Atom<Workspace> = atom((get) => {
|
|
const windowData = get(windowDataAtom);
|
|
if (windowData == null) {
|
|
return null;
|
|
}
|
|
return WOS.getObjectValue(WOS.makeORef("workspace", windowData.workspaceid), get);
|
|
});
|
|
const fullConfigAtom = atom(null) as PrimitiveAtom<FullConfigType>;
|
|
const settingsAtom = atom((get) => {
|
|
return get(fullConfigAtom)?.settings ?? {};
|
|
}) as Atom<SettingsType>;
|
|
const tabAtom: Atom<Tab> = atom((get) => {
|
|
const windowData = get(windowDataAtom);
|
|
if (windowData == null) {
|
|
return null;
|
|
}
|
|
return WOS.getObjectValue(WOS.makeORef("tab", windowData.activetabid), get);
|
|
});
|
|
const activeTabIdAtom: Atom<string> = atom((get) => {
|
|
const windowData = get(windowDataAtom);
|
|
if (windowData == null) {
|
|
return null;
|
|
}
|
|
return windowData.activetabid;
|
|
});
|
|
const controlShiftDelayAtom = atom(false);
|
|
const updaterStatusAtom = atom<UpdaterStatus>("up-to-date") as PrimitiveAtom<UpdaterStatus>;
|
|
try {
|
|
globalStore.set(updaterStatusAtom, getApi().getUpdaterStatus());
|
|
getApi().onUpdaterStatusChange((status) => {
|
|
globalStore.set(updaterStatusAtom, status);
|
|
});
|
|
} catch (_) {
|
|
// do nothing
|
|
}
|
|
|
|
const reducedMotionSettingAtom = atom((get) => get(settingsAtom)?.["window:reducedmotion"]);
|
|
const reducedMotionSystemPreferenceAtom = atom(false);
|
|
|
|
// Composite of the prefers-reduced-motion media query and the window:reducedmotion user setting.
|
|
const prefersReducedMotionAtom = atom((get) => {
|
|
const reducedMotionSetting = get(reducedMotionSettingAtom);
|
|
const reducedMotionSystemPreference = get(reducedMotionSystemPreferenceAtom);
|
|
return reducedMotionSetting || reducedMotionSystemPreference;
|
|
});
|
|
|
|
// Set up a handler for changes to the prefers-reduced-motion media query.
|
|
if (globalThis.window != null) {
|
|
const reducedMotionQuery = window.matchMedia("(prefers-reduced-motion: reduce)");
|
|
globalStore.set(reducedMotionSystemPreferenceAtom, !reducedMotionQuery || reducedMotionQuery.matches);
|
|
reducedMotionQuery?.addEventListener("change", () => {
|
|
globalStore.set(reducedMotionSystemPreferenceAtom, reducedMotionQuery.matches);
|
|
});
|
|
}
|
|
|
|
const typeAheadModalAtom = atom({});
|
|
const modalOpen = atom(false);
|
|
const allConnStatusAtom = atom<ConnStatus[]>((get) => {
|
|
const connStatuses = Array.from(ConnStatusMap.values()).map((atom) => get(atom));
|
|
return connStatuses;
|
|
});
|
|
const flashErrorsAtom = atom<FlashErrorType[]>([]);
|
|
atoms = {
|
|
// initialized in wave.ts (will not be null inside of application)
|
|
windowId: windowIdAtom,
|
|
clientId: clientIdAtom,
|
|
uiContext: uiContextAtom,
|
|
client: clientAtom,
|
|
waveWindow: windowDataAtom,
|
|
workspace: workspaceAtom,
|
|
fullConfigAtom,
|
|
settingsAtom,
|
|
tabAtom,
|
|
activeTabId: activeTabIdAtom,
|
|
isFullScreen: isFullScreenAtom,
|
|
controlShiftDelayAtom,
|
|
updaterStatusAtom,
|
|
prefersReducedMotionAtom,
|
|
typeAheadModalAtom,
|
|
modalOpen,
|
|
allConnStatus: allConnStatusAtom,
|
|
flashErrors: flashErrorsAtom,
|
|
};
|
|
}
|
|
|
|
function initGlobalWaveEventSubs() {
|
|
waveEventSubscribe(
|
|
{
|
|
eventType: "waveobj:update",
|
|
handler: (event) => {
|
|
// console.log("waveobj:update wave event handler", event);
|
|
const update: WaveObjUpdate = event.data;
|
|
WOS.updateWaveObject(update);
|
|
},
|
|
},
|
|
{
|
|
eventType: "config",
|
|
handler: (event) => {
|
|
// console.log("config wave event handler", event);
|
|
const fullConfig = (event.data as WatcherUpdate).fullconfig;
|
|
globalStore.set(atoms.fullConfigAtom, fullConfig);
|
|
},
|
|
},
|
|
{
|
|
eventType: "userinput",
|
|
handler: (event) => {
|
|
// console.log("userinput event handler", event);
|
|
const data: UserInputRequest = event.data;
|
|
modalsModel.pushModal("UserInputModal", { ...data });
|
|
},
|
|
},
|
|
{
|
|
eventType: "blockfile",
|
|
handler: (event) => {
|
|
// console.log("blockfile event update", event);
|
|
const fileData: WSFileEventData = event.data;
|
|
const fileSubject = getFileSubject(fileData.zoneid, fileData.filename);
|
|
if (fileSubject != null) {
|
|
fileSubject.next(fileData);
|
|
}
|
|
},
|
|
}
|
|
);
|
|
}
|
|
|
|
const blockCache = new Map<string, Map<string, any>>();
|
|
|
|
function useBlockCache<T>(blockId: string, name: string, makeFn: () => T): T {
|
|
let blockMap = blockCache.get(blockId);
|
|
if (blockMap == null) {
|
|
blockMap = new Map<string, any>();
|
|
blockCache.set(blockId, blockMap);
|
|
}
|
|
let value = blockMap.get(name);
|
|
if (value == null) {
|
|
value = makeFn();
|
|
blockMap.set(name, value);
|
|
}
|
|
return value as T;
|
|
}
|
|
|
|
const settingsAtomCache = new Map<string, Atom<any>>();
|
|
|
|
function getSettingsKeyAtom<T extends keyof SettingsType>(key: T): Atom<SettingsType[T]> {
|
|
let settingsKeyAtom = settingsAtomCache.get(key) as Atom<SettingsType[T]>;
|
|
if (settingsKeyAtom == null) {
|
|
settingsKeyAtom = atom((get) => {
|
|
const settings = get(atoms.settingsAtom);
|
|
if (settings == null) {
|
|
return null;
|
|
}
|
|
return settings[key];
|
|
});
|
|
settingsAtomCache.set(key, settingsKeyAtom);
|
|
}
|
|
return settingsKeyAtom;
|
|
}
|
|
|
|
function useSettingsPrefixAtom(prefix: string): Atom<SettingsType> {
|
|
// TODO: use a shallow equal here to make this more efficient
|
|
let settingsPrefixAtom = settingsAtomCache.get(prefix + ":") as Atom<SettingsType>;
|
|
if (settingsPrefixAtom == null) {
|
|
settingsPrefixAtom = atom((get) => {
|
|
const settings = get(atoms.settingsAtom);
|
|
if (settings == null) {
|
|
return {};
|
|
}
|
|
return getPrefixedSettings(settings, prefix);
|
|
});
|
|
settingsAtomCache.set(prefix + ":", settingsPrefixAtom);
|
|
}
|
|
return settingsPrefixAtom;
|
|
}
|
|
|
|
const blockAtomCache = new Map<string, Map<string, Atom<any>>>();
|
|
|
|
function useBlockAtom<T>(blockId: string, name: string, makeFn: () => Atom<T>): Atom<T> {
|
|
let blockCache = blockAtomCache.get(blockId);
|
|
if (blockCache == null) {
|
|
blockCache = new Map<string, Atom<any>>();
|
|
blockAtomCache.set(blockId, blockCache);
|
|
}
|
|
let atom = blockCache.get(name);
|
|
if (atom == null) {
|
|
atom = makeFn();
|
|
blockCache.set(name, atom);
|
|
console.log("New BlockAtom", blockId, name);
|
|
}
|
|
return atom as Atom<T>;
|
|
}
|
|
|
|
function useBlockDataLoaded(blockId: string): boolean {
|
|
const loadedAtom = useBlockAtom<boolean>(blockId, "block-loaded", () => {
|
|
return WOS.getWaveObjectLoadingAtom(WOS.makeORef("block", blockId));
|
|
});
|
|
return useAtomValue(loadedAtom);
|
|
}
|
|
|
|
/**
|
|
* Get the preload api.
|
|
*/
|
|
function getApi(): ElectronApi {
|
|
return (window as any).api;
|
|
}
|
|
|
|
async function createBlock(blockDef: BlockDef, magnified = false): Promise<string> {
|
|
const rtOpts: RuntimeOpts = { termsize: { rows: 25, cols: 80 } };
|
|
const blockId = await ObjectService.CreateBlock(blockDef, rtOpts);
|
|
const insertNodeAction: LayoutTreeInsertNodeAction = {
|
|
type: LayoutTreeActionType.InsertNode,
|
|
node: newLayoutNode(undefined, undefined, undefined, { blockId }),
|
|
magnified,
|
|
focused: true,
|
|
};
|
|
const activeTabId = globalStore.get(atoms.uiContext).activetabid;
|
|
const layoutModel = getLayoutModelForTabById(activeTabId);
|
|
layoutModel.treeReducer(insertNodeAction);
|
|
return blockId;
|
|
}
|
|
|
|
// when file is not found, returns {data: null, fileInfo: null}
|
|
async function fetchWaveFile(
|
|
zoneId: string,
|
|
fileName: string,
|
|
offset?: number
|
|
): Promise<{ data: Uint8Array; fileInfo: WaveFile }> {
|
|
const usp = new URLSearchParams();
|
|
usp.set("zoneid", zoneId);
|
|
usp.set("name", fileName);
|
|
if (offset != null) {
|
|
usp.set("offset", offset.toString());
|
|
}
|
|
const resp = await fetch(getWebServerEndpoint() + "/wave/file?" + usp.toString());
|
|
if (!resp.ok) {
|
|
if (resp.status === 404) {
|
|
return { data: null, fileInfo: null };
|
|
}
|
|
throw new Error("error getting wave file: " + resp.statusText);
|
|
}
|
|
if (resp.status == 204) {
|
|
return { data: null, fileInfo: null };
|
|
}
|
|
const fileInfo64 = resp.headers.get("X-ZoneFileInfo");
|
|
if (fileInfo64 == null) {
|
|
throw new Error(`missing zone file info for ${zoneId}:${fileName}`);
|
|
}
|
|
const fileInfo = JSON.parse(atob(fileInfo64));
|
|
const data = await resp.arrayBuffer();
|
|
return { data: new Uint8Array(data), fileInfo };
|
|
}
|
|
|
|
function setNodeFocus(nodeId: string) {
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
layoutModel.focusNode(nodeId);
|
|
}
|
|
|
|
const objectIdWeakMap = new WeakMap();
|
|
let objectIdCounter = 0;
|
|
function getObjectId(obj: any): number {
|
|
if (!objectIdWeakMap.has(obj)) {
|
|
objectIdWeakMap.set(obj, objectIdCounter++);
|
|
}
|
|
return objectIdWeakMap.get(obj);
|
|
}
|
|
|
|
let cachedIsDev: boolean = null;
|
|
|
|
function isDev() {
|
|
if (cachedIsDev == null) {
|
|
cachedIsDev = getApi().getIsDev();
|
|
}
|
|
return cachedIsDev;
|
|
}
|
|
|
|
let cachedUserName: string = null;
|
|
|
|
function getUserName(): string {
|
|
if (cachedUserName == null) {
|
|
cachedUserName = getApi().getUserName();
|
|
}
|
|
return cachedUserName;
|
|
}
|
|
|
|
let cachedHostName: string = null;
|
|
|
|
function getHostName(): string {
|
|
if (cachedHostName == null) {
|
|
cachedHostName = getApi().getHostName();
|
|
}
|
|
return cachedHostName;
|
|
}
|
|
|
|
/**
|
|
* Open a link in a new window, or in a new web widget. The user can set all links to open in a new web widget using the `web:openlinksinternally` setting.
|
|
* @param uri The link to open.
|
|
* @param forceOpenInternally Force the link to open in a new web widget.
|
|
*/
|
|
async function openLink(uri: string, forceOpenInternally = false) {
|
|
if (forceOpenInternally || globalStore.get(atoms.settingsAtom)?.["web:openlinksinternally"]) {
|
|
const blockDef: BlockDef = {
|
|
meta: {
|
|
view: "web",
|
|
url: uri,
|
|
},
|
|
};
|
|
await createBlock(blockDef);
|
|
} else {
|
|
getApi().openExternal(uri);
|
|
}
|
|
}
|
|
|
|
function registerBlockComponentModel(blockId: string, bcm: BlockComponentModel) {
|
|
blockComponentModelMap.set(blockId, bcm);
|
|
}
|
|
|
|
function unregisterBlockComponentModel(blockId: string) {
|
|
blockComponentModelMap.delete(blockId);
|
|
}
|
|
|
|
function getBlockComponentModel(blockId: string): BlockComponentModel {
|
|
return blockComponentModelMap.get(blockId);
|
|
}
|
|
|
|
function refocusNode(blockId: string) {
|
|
if (blockId == null) {
|
|
return;
|
|
}
|
|
const layoutModel = getLayoutModelForActiveTab();
|
|
const layoutNodeId = layoutModel.getNodeByBlockId(blockId);
|
|
if (layoutNodeId?.id == null) {
|
|
return;
|
|
}
|
|
layoutModel.focusNode(layoutNodeId.id);
|
|
const bcm = getBlockComponentModel(blockId);
|
|
const ok = bcm?.viewModel?.giveFocus?.();
|
|
if (!ok) {
|
|
const inputElem = document.getElementById(`${blockId}-dummy-focus`);
|
|
inputElem?.focus();
|
|
}
|
|
}
|
|
|
|
function countersClear() {
|
|
Counters.clear();
|
|
}
|
|
|
|
function counterInc(name: string, incAmt: number = 1) {
|
|
let count = Counters.get(name) ?? 0;
|
|
count += incAmt;
|
|
Counters.set(name, count);
|
|
}
|
|
|
|
function countersPrint() {
|
|
let outStr = "";
|
|
for (const [name, count] of Counters.entries()) {
|
|
outStr += `${name}: ${count}\n`;
|
|
}
|
|
console.log(outStr);
|
|
}
|
|
|
|
async function loadConnStatus() {
|
|
const connStatusArr = await ClientService.GetAllConnStatus();
|
|
if (connStatusArr == null) {
|
|
return;
|
|
}
|
|
for (const connStatus of connStatusArr) {
|
|
const curAtom = getConnStatusAtom(connStatus.connection);
|
|
globalStore.set(curAtom, connStatus);
|
|
}
|
|
}
|
|
|
|
function subscribeToConnEvents() {
|
|
waveEventSubscribe({
|
|
eventType: "connchange",
|
|
handler: (event: WaveEvent) => {
|
|
try {
|
|
const connStatus = event.data as ConnStatus;
|
|
if (connStatus == null || isBlank(connStatus.connection)) {
|
|
return;
|
|
}
|
|
console.log("connstatus update", connStatus);
|
|
let curAtom = getConnStatusAtom(connStatus.connection);
|
|
globalStore.set(curAtom, connStatus);
|
|
} catch (e) {
|
|
console.log("connchange error", e);
|
|
}
|
|
},
|
|
});
|
|
}
|
|
|
|
function getConnStatusAtom(conn: string): PrimitiveAtom<ConnStatus> {
|
|
let rtn = ConnStatusMap.get(conn);
|
|
if (rtn == null) {
|
|
if (isBlank(conn)) {
|
|
// create a fake "local" status atom that's always connected
|
|
const connStatus: ConnStatus = {
|
|
connection: conn,
|
|
connected: true,
|
|
error: null,
|
|
status: "connected",
|
|
hasconnected: true,
|
|
activeconnnum: 0,
|
|
};
|
|
rtn = atom(connStatus);
|
|
} else {
|
|
const connStatus: ConnStatus = {
|
|
connection: conn,
|
|
connected: false,
|
|
error: null,
|
|
status: "disconnected",
|
|
hasconnected: false,
|
|
activeconnnum: 0,
|
|
};
|
|
rtn = atom(connStatus);
|
|
}
|
|
ConnStatusMap.set(conn, rtn);
|
|
}
|
|
return rtn;
|
|
}
|
|
|
|
function pushFlashError(ferr: FlashErrorType) {
|
|
if (ferr.expiration == null) {
|
|
ferr.expiration = Date.now() + 5000;
|
|
}
|
|
ferr.id = crypto.randomUUID();
|
|
globalStore.set(atoms.flashErrors, (prev) => {
|
|
return [...prev, ferr];
|
|
});
|
|
}
|
|
|
|
function removeFlashError(id: string) {
|
|
globalStore.set(atoms.flashErrors, (prev) => {
|
|
return prev.filter((ferr) => ferr.id !== id);
|
|
});
|
|
}
|
|
|
|
export {
|
|
atoms,
|
|
counterInc,
|
|
countersClear,
|
|
countersPrint,
|
|
createBlock,
|
|
fetchWaveFile,
|
|
getApi,
|
|
getBlockComponentModel,
|
|
getConnStatusAtom,
|
|
getHostName,
|
|
getObjectId,
|
|
getSettingsKeyAtom,
|
|
getUserName,
|
|
globalStore,
|
|
initGlobal,
|
|
initGlobalWaveEventSubs,
|
|
isDev,
|
|
loadConnStatus,
|
|
openLink,
|
|
PLATFORM,
|
|
pushFlashError,
|
|
refocusNode,
|
|
registerBlockComponentModel,
|
|
removeFlashError,
|
|
setNodeFocus,
|
|
setPlatform,
|
|
subscribeToConnEvents,
|
|
unregisterBlockComponentModel,
|
|
useBlockAtom,
|
|
useBlockCache,
|
|
useBlockDataLoaded,
|
|
useSettingsPrefixAtom,
|
|
WOS,
|
|
};
|