// Copyright 2024, Command Line Inc. // SPDX-License-Identifier: Apache-2.0 import { blockViewToIcon, blockViewToName, computeConnColorNum, ConnectionButton, getBlockHeaderIcon, Input, } from "@/app/block/blockutil"; import { Button } from "@/app/element/button"; import { useDimensionsWithCallbackRef } from "@/app/hook/useDimensions"; import { TypeAheadModal } from "@/app/modals/typeaheadmodal"; import { ContextMenuModel } from "@/app/store/contextmenu"; import { atoms, createBlock, getApi, getBlockComponentModel, getConnStatusAtom, getHostName, getSettingsKeyAtom, getUserName, globalStore, refocusNode, useBlockAtom, WOS, } from "@/app/store/global"; import { RpcApi } from "@/app/store/wshclientapi"; import { TabRpcClient } from "@/app/store/wshrpcutil"; import { ErrorBoundary } from "@/element/errorboundary"; import { IconButton } from "@/element/iconbutton"; import { MagnifyIcon } from "@/element/magnify"; import { MenuButton } from "@/element/menubutton"; import { NodeModel } from "@/layout/index"; import * as keyutil from "@/util/keyutil"; import * as util from "@/util/util"; import clsx from "clsx"; import * as jotai from "jotai"; import * as React from "react"; import { BlockFrameProps } from "./blocktypes"; const NumActiveConnColors = 8; function handleHeaderContextMenu( e: React.MouseEvent, blockData: Block, viewModel: ViewModel, magnified: boolean, onMagnifyToggle: () => void, onClose: () => void ) { e.preventDefault(); e.stopPropagation(); let menu: ContextMenuItem[] = [ { label: magnified ? "Un-Magnify Block" : "Magnify Block", click: () => { onMagnifyToggle(); }, }, // { // label: "Move to New Window", // click: () => { // const currentTabId = globalStore.get(atoms.staticTabId); // try { // services.WindowService.MoveBlockToNewWindow(currentTabId, blockData.oid); // } catch (e) { // console.error("error moving block to new window", e); // } // }, // }, { type: "separator" }, { label: "Copy BlockId", click: () => { navigator.clipboard.writeText(blockData.oid); }, }, ]; const extraItems = viewModel?.getSettingsMenuItems?.(); if (extraItems && extraItems.length > 0) menu.push({ type: "separator" }, ...extraItems); menu.push( { type: "separator" }, { label: "Close Block", click: onClose, } ); ContextMenuModel.showContextMenu(menu, e); } function getViewIconElem(viewIconUnion: string | IconButtonDecl, blockData: Block): JSX.Element { if (viewIconUnion == null || typeof viewIconUnion === "string") { const viewIcon = viewIconUnion as string; return
{getBlockHeaderIcon(viewIcon, blockData)}
; } else { return ; } } const OptMagnifyButton = React.memo( ({ magnified, toggleMagnify, disabled }: { magnified: boolean; toggleMagnify: () => void; disabled: boolean }) => { const magnifyDecl: IconButtonDecl = { elemtype: "iconbutton", icon: , title: magnified ? "Minimize" : "Magnify", click: toggleMagnify, disabled, }; return ; } ); function computeEndIcons( viewModel: ViewModel, nodeModel: NodeModel, onContextMenu: (e: React.MouseEvent) => void ): JSX.Element[] { const endIconsElem: JSX.Element[] = []; const endIconButtons = util.useAtomValueSafe(viewModel?.endIconButtons); const magnified = jotai.useAtomValue(nodeModel.isMagnified); const ephemeral = jotai.useAtomValue(nodeModel.isEphemeral); const numLeafs = jotai.useAtomValue(nodeModel.numLeafs); const magnifyDisabled = numLeafs <= 1; if (endIconButtons && endIconButtons.length > 0) { endIconsElem.push(...endIconButtons.map((button, idx) => )); } const settingsDecl: IconButtonDecl = { elemtype: "iconbutton", icon: "cog", title: "Settings", click: onContextMenu, }; endIconsElem.push(); if (ephemeral) { const addToLayoutDecl: IconButtonDecl = { elemtype: "iconbutton", icon: "circle-plus", title: "Add to Layout", click: () => { nodeModel.addEphemeralNodeToLayout(); }, }; endIconsElem.push(); } else { endIconsElem.push( ); } const closeDecl: IconButtonDecl = { elemtype: "iconbutton", icon: "xmark-large", title: "Close", click: nodeModel.onClose, }; endIconsElem.push(); return endIconsElem; } const BlockFrame_Header = ({ nodeModel, viewModel, preview, connBtnRef, changeConnModalAtom, error, }: BlockFrameProps & { changeConnModalAtom: jotai.PrimitiveAtom; error?: Error }) => { const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", nodeModel.blockId)); let viewName = util.useAtomValueSafe(viewModel?.viewName) ?? blockViewToName(blockData?.meta?.view); const showBlockIds = jotai.useAtomValue(getSettingsKeyAtom("blockheader:showblockids")); let viewIconUnion = util.useAtomValueSafe(viewModel?.viewIcon) ?? blockViewToIcon(blockData?.meta?.view); const preIconButton = util.useAtomValueSafe(viewModel?.preIconButton); let headerTextUnion = util.useAtomValueSafe(viewModel?.viewText); const magnified = jotai.useAtomValue(nodeModel.isMagnified); const prevMagifiedState = React.useRef(magnified); const manageConnection = util.useAtomValueSafe(viewModel?.manageConnection); const dragHandleRef = preview ? null : nodeModel.dragHandleRef; const connName = blockData?.meta?.connection; const connStatus = util.useAtomValueSafe(getConnStatusAtom(connName)); const wshProblem = connName && !connStatus?.wshenabled && connStatus?.status == "connected"; React.useEffect(() => { if (!magnified || preview || prevMagifiedState.current) { return; } RpcApi.ActivityCommand(TabRpcClient, { nummagnify: 1 }); }, [magnified]); if (blockData?.meta?.["frame:title"]) { viewName = blockData.meta["frame:title"]; } if (blockData?.meta?.["frame:icon"]) { viewIconUnion = blockData.meta["frame:icon"]; } if (blockData?.meta?.["frame:text"]) { headerTextUnion = blockData.meta["frame:text"]; } const onContextMenu = React.useCallback( (e: React.MouseEvent) => { handleHeaderContextMenu(e, blockData, viewModel, magnified, nodeModel.toggleMagnify, nodeModel.onClose); }, [magnified] ); const endIconsElem = computeEndIcons(viewModel, nodeModel, onContextMenu); const viewIconElem = getViewIconElem(viewIconUnion, blockData); let preIconButtonElem: JSX.Element = null; if (preIconButton) { preIconButtonElem = ; } const headerTextElems: JSX.Element[] = []; if (typeof headerTextUnion === "string") { if (!util.isBlank(headerTextUnion)) { headerTextElems.push(
‎{headerTextUnion}
); } } else if (Array.isArray(headerTextUnion)) { headerTextElems.push(...renderHeaderElements(headerTextUnion, preview)); } if (error != null) { const copyHeaderErr = () => { navigator.clipboard.writeText(error.message + "\n" + error.stack); }; headerTextElems.push(
); } const wshInstallButton: IconButtonDecl = { elemtype: "iconbutton", icon: "link-slash", title: "wsh is not installed for this connection", }; return (
{preIconButtonElem}
{viewIconElem}
{viewName}
{showBlockIds &&
[{nodeModel.blockId.substring(0, 8)}]
}
{manageConnection && ( )} {manageConnection && wshProblem && ( )}
{headerTextElems}
{endIconsElem}
); }; const HeaderTextElem = React.memo(({ elem, preview }: { elem: HeaderElem; preview: boolean }) => { if (elem.elemtype == "iconbutton") { return ; } else if (elem.elemtype == "input") { return ; } else if (elem.elemtype == "text") { return (
elem?.onClick(e)}> ‎{elem.text}
); } else if (elem.elemtype == "textbutton") { return ( ); } else if (elem.elemtype == "div") { return (
{elem.children.map((child, childIdx) => ( ))}
); } else if (elem.elemtype == "menubutton") { return ; } return null; }); function renderHeaderElements(headerTextUnion: HeaderElem[], preview: boolean): JSX.Element[] { const headerTextElems: JSX.Element[] = []; for (let idx = 0; idx < headerTextUnion.length; idx++) { const elem = headerTextUnion[idx]; const renderedElement = ; if (renderedElement) { headerTextElems.push(renderedElement); } } return headerTextElems; } const ConnStatusOverlay = React.memo( ({ nodeModel, viewModel, changeConnModalAtom, }: { nodeModel: NodeModel; viewModel: ViewModel; changeConnModalAtom: jotai.PrimitiveAtom; }) => { const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", nodeModel.blockId)); const [connModalOpen] = jotai.useAtom(changeConnModalAtom); const connName = blockData.meta?.connection; const connStatus = jotai.useAtomValue(getConnStatusAtom(connName)); const isLayoutMode = jotai.useAtomValue(atoms.controlShiftDelayAtom); const [overlayRefCallback, _, domRect] = useDimensionsWithCallbackRef(30); const width = domRect?.width; const [showError, setShowError] = React.useState(false); const fullConfig = jotai.useAtomValue(atoms.fullConfigAtom); const [showWshError, setShowWshError] = React.useState(false); React.useEffect(() => { if (width) { const hasError = !util.isBlank(connStatus.error); const showError = hasError && width >= 250 && connStatus.status != "connecting"; setShowError(showError); } }, [width, connStatus, setShowError]); const handleTryReconnect = React.useCallback(() => { const prtn = RpcApi.ConnConnectCommand(TabRpcClient, { host: connName }, { timeout: 60000 }); prtn.catch((e) => console.log("error reconnecting", connName, e)); }, [connName]); const handleDisableWsh = React.useCallback(async () => { // using unknown is a hack. we need proper types for the // connection config on the frontend const metamaptype: unknown = { "conn:wshenabled": false, }; const data: ConnConfigRequest = { host: connName, metamaptype: metamaptype, }; try { await RpcApi.SetConnectionsConfigCommand(TabRpcClient, data); } catch (e) { console.log("problem setting connection config: ", e); } }, [connName]); const handleRemoveWshError = React.useCallback(async () => { try { await RpcApi.DismissWshFailCommand(TabRpcClient, connName); } catch (e) { console.log("unable to dismiss wsh error: ", e); } }, [connName]); let statusText = `Disconnected from "${connName}"`; let showReconnect = true; if (connStatus.status == "connecting") { statusText = `Connecting to "${connName}"...`; showReconnect = false; } if (connStatus.status == "connected") { showReconnect = false; } let reconDisplay = null; let reconClassName = "outlined grey"; if (width && width < 350) { reconDisplay = ; reconClassName = clsx(reconClassName, "font-size-12 vertical-padding-5 horizontal-padding-6"); } else { reconDisplay = "Reconnect"; reconClassName = clsx(reconClassName, "font-size-11 vertical-padding-3 horizontal-padding-7"); } const showIcon = connStatus.status != "connecting"; const wshConfigEnabled = fullConfig?.connections?.[connName]?.["conn:wshenabled"] ?? true; React.useEffect(() => { const showWshErrorTemp = connStatus.status == "connected" && connStatus.wsherror && connStatus.wsherror != "" && wshConfigEnabled; setShowWshError(showWshErrorTemp); }, [connStatus, wshConfigEnabled]); if (!showWshError && (isLayoutMode || connStatus.status == "connected" || connModalOpen)) { return null; } return (
{showIcon && }
{statusText}
{showError ?
error: {connStatus.error}
: null} {showWshError ? (
unable to use wsh: {connStatus.wsherror}
) : null} {showWshError && ( )}
{showReconnect ? (
) : null} {showWshError ? (
) : null}
); } ); const BlockMask = React.memo(({ nodeModel }: { nodeModel: NodeModel }) => { const isFocused = jotai.useAtomValue(nodeModel.isFocused); const blockNum = jotai.useAtomValue(nodeModel.blockNum); const isLayoutMode = jotai.useAtomValue(atoms.controlShiftDelayAtom); const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", nodeModel.blockId)); const style: React.CSSProperties = {}; let showBlockMask = false; if (isFocused) { const tabData = jotai.useAtomValue(atoms.tabAtom); const tabActiveBorderColor = tabData?.meta?.["bg:activebordercolor"]; if (tabActiveBorderColor) { style.borderColor = tabActiveBorderColor; } if (blockData?.meta?.["frame:activebordercolor"]) { style.borderColor = blockData.meta["frame:activebordercolor"]; } } else { const tabData = jotai.useAtomValue(atoms.tabAtom); const tabBorderColor = tabData?.meta?.["bg:bordercolor"]; if (tabBorderColor) { style.borderColor = tabBorderColor; } if (blockData?.meta?.["frame:bordercolor"]) { style.borderColor = blockData.meta["frame:bordercolor"]; } } let innerElem = null; if (isLayoutMode) { showBlockMask = true; innerElem = (
{blockNum}
); } return (
{innerElem}
); }); const BlockFrame_Default_Component = (props: BlockFrameProps) => { const { nodeModel, viewModel, blockModel, preview, numBlocksInTab, children } = props; const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", nodeModel.blockId)); const isFocused = jotai.useAtomValue(nodeModel.isFocused); const viewIconUnion = util.useAtomValueSafe(viewModel?.viewIcon) ?? blockViewToIcon(blockData?.meta?.view); const customBg = util.useAtomValueSafe(viewModel?.blockBg); const manageConnection = util.useAtomValueSafe(viewModel?.manageConnection); const changeConnModalAtom = useBlockAtom(nodeModel.blockId, "changeConn", () => { return jotai.atom(false); }) as jotai.PrimitiveAtom; const connModalOpen = jotai.useAtomValue(changeConnModalAtom); const isMagnified = jotai.useAtomValue(nodeModel.isMagnified); const isEphemeral = jotai.useAtomValue(nodeModel.isEphemeral); const [magnifiedBlockBlurAtom] = React.useState(() => getSettingsKeyAtom("window:magnifiedblockblurprimarypx")); const magnifiedBlockBlur = jotai.useAtomValue(magnifiedBlockBlurAtom); const [magnifiedBlockOpacityAtom] = React.useState(() => getSettingsKeyAtom("window:magnifiedblockopacity")); const magnifiedBlockOpacity = jotai.useAtomValue(magnifiedBlockOpacityAtom); const connBtnRef = React.useRef(); React.useEffect(() => { if (!manageConnection) { return; } const bcm = getBlockComponentModel(nodeModel.blockId); if (bcm != null) { bcm.openSwitchConnection = () => { globalStore.set(changeConnModalAtom, true); }; } return () => { const bcm = getBlockComponentModel(nodeModel.blockId); if (bcm != null) { bcm.openSwitchConnection = null; } }; }, [manageConnection]); React.useEffect(() => { // on mount, if manageConnection, call ConnEnsure if (!manageConnection || blockData == null || preview) { return; } const connName = blockData?.meta?.connection; if (!util.isBlank(connName)) { console.log("ensure conn", nodeModel.blockId, connName); RpcApi.ConnEnsureCommand(TabRpcClient, connName, { timeout: 60000 }).catch((e) => { console.log("error ensuring connection", nodeModel.blockId, connName, e); }); } }, [manageConnection, blockData]); const viewIconElem = getViewIconElem(viewIconUnion, blockData); const innerStyle: React.CSSProperties = {}; if (!preview && customBg?.bg != null) { innerStyle.background = customBg.bg; if (customBg["bg:opacity"] != null) { innerStyle.opacity = customBg["bg:opacity"]; } if (customBg["bg:blendmode"] != null) { innerStyle.backgroundBlendMode = customBg["bg:blendmode"]; } } const previewElem =
{viewIconElem}
; const headerElem = ( ); const headerElemNoView = React.cloneElement(headerElem, { viewModel: null }); return (
{preview || viewModel == null ? null : ( )}
{headerElem} {preview ? previewElem : children}
{preview || viewModel == null || !connModalOpen ? null : ( )}
); }; const ChangeConnectionBlockModal = React.memo( ({ blockId, viewModel, blockRef, connBtnRef, changeConnModalAtom, nodeModel, }: { blockId: string; viewModel: ViewModel; blockRef: React.RefObject; connBtnRef: React.RefObject; changeConnModalAtom: jotai.PrimitiveAtom; nodeModel: NodeModel; }) => { const [connSelected, setConnSelected] = React.useState(""); const changeConnModalOpen = jotai.useAtomValue(changeConnModalAtom); const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", blockId)); const isNodeFocused = jotai.useAtomValue(nodeModel.isFocused); const connection = blockData?.meta?.connection; const connStatusAtom = getConnStatusAtom(connection); const connStatus = jotai.useAtomValue(connStatusAtom); const [connList, setConnList] = React.useState>([]); const [wslList, setWslList] = React.useState>([]); const allConnStatus = jotai.useAtomValue(atoms.allConnStatus); const [rowIndex, setRowIndex] = React.useState(0); const connStatusMap = new Map(); const fullConfig = jotai.useAtomValue(atoms.fullConfigAtom); const connectionsConfig = fullConfig.connections; let filterOutNowsh = util.useAtomValueSafe(viewModel.filterOutNowsh) ?? true; let maxActiveConnNum = 1; for (const conn of allConnStatus) { if (conn.activeconnnum > maxActiveConnNum) { maxActiveConnNum = conn.activeconnnum; } connStatusMap.set(conn.connection, conn); } React.useEffect(() => { if (!changeConnModalOpen) { setConnList([]); return; } const prtn = RpcApi.ConnListCommand(TabRpcClient, { timeout: 2000 }); prtn.then((newConnList) => { setConnList(newConnList ?? []); }).catch((e) => console.log("unable to load conn list from backend. using blank list: ", e)); const p2rtn = RpcApi.WslListCommand(TabRpcClient, { timeout: 2000 }); p2rtn .then((newWslList) => { console.log(newWslList); setWslList(newWslList ?? []); }) .catch((e) => { // removing this log and failing silentyly since it will happen // if a system isn't using the wsl. and would happen every time the // typeahead was opened. good candidate for verbose log level. //console.log("unable to load wsl list from backend. using blank list: ", e) }); }, [changeConnModalOpen, setConnList]); const changeConnection = React.useCallback( async (connName: string) => { if (connName == "") { connName = null; } if (connName == blockData?.meta?.connection) { return; } const oldCwd = blockData?.meta?.file ?? ""; let newCwd: string; if (oldCwd == "") { newCwd = ""; } else { newCwd = "~"; } await RpcApi.SetMetaCommand(TabRpcClient, { oref: WOS.makeORef("block", blockId), meta: { connection: connName, file: newCwd }, }); try { await RpcApi.ConnEnsureCommand(TabRpcClient, connName, { timeout: 60000 }); } catch (e) { console.log("error connecting", blockId, connName, e); } }, [blockId, blockData] ); let createNew: boolean = true; let showReconnect: boolean = true; if (connSelected == "") { createNew = false; } else { showReconnect = false; } const filteredList: Array = []; for (const conn of connList) { if ( conn.includes(connSelected) && connectionsConfig?.[conn]?.["display:hidden"] != true && (connectionsConfig?.[conn]?.["conn:wshenabled"] != false || !filterOutNowsh) // != false is necessary because of defaults ) { filteredList.push(conn); if (conn === connSelected) { createNew = false; } } } const filteredWslList: Array = []; for (const conn of wslList) { if ( conn.includes(connSelected) && connectionsConfig?.[conn]?.["display:hidden"] != true && (connectionsConfig?.[conn]?.["conn:wshenabled"] != false || !filterOutNowsh) // != false is necessary because of defaults ) { filteredWslList.push(conn); if (conn === connSelected) { createNew = false; } } } // priority handles special suggestions when necessary // for instance, when reconnecting const newConnectionSuggestion: SuggestionConnectionItem = { status: "connected", icon: "plus", iconColor: "var(--grey-text-color)", label: `${connSelected} (New Connection)`, value: "", onSelect: (_: string) => { changeConnection(connSelected); globalStore.set(changeConnModalAtom, false); }, }; const reconnectSuggestion: SuggestionConnectionItem = { status: "connected", icon: "arrow-right-arrow-left", iconColor: "var(--grey-text-color)", label: `Reconnect to ${connStatus.connection}`, value: "", onSelect: async (_: string) => { const prtn = RpcApi.ConnConnectCommand( TabRpcClient, { host: connStatus.connection }, { timeout: 60000 } ); prtn.catch((e) => console.log("error reconnecting", connStatus.connection, e)); }, }; const localName = getUserName() + "@" + getHostName(); const localSuggestion: SuggestionConnectionScope = { headerText: "Local", items: [], }; if (localName.includes(connSelected)) { localSuggestion.items.push({ status: "connected", icon: "laptop", iconColor: "var(--grey-text-color)", value: "", label: localName, current: connection == null, }); } if (localName == connSelected) { createNew = false; } for (const wslConn of filteredWslList) { const connStatus = connStatusMap.get(wslConn); const connColorNum = computeConnColorNum(connStatus); localSuggestion.items.push({ status: "connected", icon: "arrow-right-arrow-left", iconColor: connStatus?.status == "connected" ? `var(--conn-icon-color-${connColorNum})` : "var(--grey-text-color)", value: "wsl://" + wslConn, label: "wsl://" + wslConn, current: "wsl://" + wslConn == connection, }); } const remoteItems = filteredList.map((connName) => { const connStatus = connStatusMap.get(connName); const connColorNum = computeConnColorNum(connStatus); const item: SuggestionConnectionItem = { status: "connected", icon: "arrow-right-arrow-left", iconColor: connStatus?.status == "connected" ? `var(--conn-icon-color-${connColorNum})` : "var(--grey-text-color)", value: connName, label: connName, current: connName == connection, }; return item; }); const connectionsEditItem: SuggestionConnectionItem = { status: "disconnected", icon: "gear", iconColor: "var(--grey-text-color", value: "Edit Connections", label: "Edit Connections", onSelect: () => { util.fireAndForget(async () => { globalStore.set(changeConnModalAtom, false); const path = `${getApi().getConfigDir()}/connections.json`; const blockDef: BlockDef = { meta: { view: "preview", file: path, }, }; await createBlock(blockDef, false, true); }); }, }; const sortedRemoteItems = remoteItems.sort( (itemA: SuggestionConnectionItem, itemB: SuggestionConnectionItem) => { const connNameA = itemA.value; const connNameB = itemB.value; const valueA = connectionsConfig?.[connNameA]?.["display:order"] ?? 0; const valueB = connectionsConfig?.[connNameB]?.["display:order"] ?? 0; return valueA - valueB; } ); const remoteSuggestions: SuggestionConnectionScope = { headerText: "Remote", items: [...sortedRemoteItems], }; const suggestions: Array = [ ...(showReconnect && (connStatus.status == "disconnected" || connStatus.status == "error") ? [reconnectSuggestion] : []), ...(localSuggestion.items.length > 0 ? [localSuggestion] : []), ...(remoteSuggestions.items.length > 0 ? [remoteSuggestions] : []), ...(connSelected == "" ? [connectionsEditItem] : []), ...(createNew ? [newConnectionSuggestion] : []), ]; let selectionList: Array = suggestions.flatMap((item) => { if ("items" in item) { return item.items; } return item; }); // quick way to change icon color when highlighted selectionList = selectionList.map((item, index) => { if (index == rowIndex && item.iconColor == "var(--grey-text-color)") { item.iconColor = "var(--main-text-color)"; } return item; }); const handleTypeAheadKeyDown = React.useCallback( (waveEvent: WaveKeyboardEvent): boolean => { if (keyutil.checkKeyPressed(waveEvent, "Enter")) { const rowItem = selectionList[rowIndex]; if ("onSelect" in rowItem && rowItem.onSelect) { rowItem.onSelect(rowItem.value); } else { changeConnection(rowItem.value); globalStore.set(changeConnModalAtom, false); } } if (keyutil.checkKeyPressed(waveEvent, "Escape")) { globalStore.set(changeConnModalAtom, false); setConnSelected(""); refocusNode(blockId); return true; } if (keyutil.checkKeyPressed(waveEvent, "ArrowUp")) { setRowIndex((idx) => Math.max(idx - 1, 0)); return true; } if (keyutil.checkKeyPressed(waveEvent, "ArrowDown")) { setRowIndex((idx) => Math.min(idx + 1, selectionList.length - 1)); return true; } setRowIndex(0); }, [changeConnModalAtom, viewModel, blockId, connSelected, selectionList] ); React.useEffect(() => { // this is specifically for the case when the list shrinks due // to a search filter setRowIndex((idx) => Math.min(idx, selectionList.flat().length - 1)); }, [selectionList, setRowIndex]); // this check was also moved to BlockFrame to prevent all the above code from running unnecessarily if (!changeConnModalOpen) { return null; } return ( { changeConnection(selected); globalStore.set(changeConnModalAtom, false); }} selectIndex={rowIndex} autoFocus={isNodeFocused} onKeyDown={(e) => keyutil.keydownWrapper(handleTypeAheadKeyDown)(e)} onChange={(current: string) => setConnSelected(current)} value={connSelected} label="Connect to (username@host)..." onClickBackdrop={() => globalStore.set(changeConnModalAtom, false)} /> ); } ); const BlockFrame_Default = React.memo(BlockFrame_Default_Component) as typeof BlockFrame_Default_Component; const BlockFrame = React.memo((props: BlockFrameProps) => { const blockId = props.nodeModel.blockId; const [blockData] = WOS.useWaveObjectValue(WOS.makeORef("block", blockId)); const tabData = jotai.useAtomValue(atoms.tabAtom); if (!blockId || !blockData) { return null; } const FrameElem = BlockFrame_Default; const numBlocks = tabData?.blockids?.length ?? 0; return ; }); export { BlockFrame, NumActiveConnColors };