2023-06-16 04:23:19 +00:00
|
|
|
import {
|
|
|
|
Close as CloseIcon,
|
2023-12-01 20:57:21 +00:00
|
|
|
Dashboard as DashboardIcon,
|
2024-03-23 03:37:39 +00:00
|
|
|
Fullscreen as FullscreenIcon,
|
2023-06-16 04:23:19 +00:00
|
|
|
Keyboard as KeyboardIcon,
|
|
|
|
} from '@mui/icons-material';
|
2023-07-20 06:56:44 +00:00
|
|
|
import { Box, Menu, styled, Typography } from '@mui/material';
|
2021-08-05 15:12:59 +00:00
|
|
|
import RFB from '@novnc/novnc/core/rfb';
|
2023-06-16 04:23:19 +00:00
|
|
|
import dynamic from 'next/dynamic';
|
2023-07-20 01:08:23 +00:00
|
|
|
import { useState, useEffect, FC, useMemo, useRef, useCallback } from 'react';
|
2023-06-16 04:23:19 +00:00
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
import IconButton from '../IconButton';
|
2021-07-12 23:45:21 +00:00
|
|
|
import keyCombinations from './keyCombinations';
|
2023-07-20 05:33:58 +00:00
|
|
|
import MenuItem from '../MenuItem';
|
2023-07-20 01:08:23 +00:00
|
|
|
import { Panel, PanelHeader } from '../Panels';
|
2023-12-13 20:38:31 +00:00
|
|
|
import ServerMenu from '../ServerMenu';
|
2021-07-19 22:34:57 +00:00
|
|
|
import Spinner from '../Spinner';
|
2024-05-22 03:16:00 +00:00
|
|
|
import { BodyText, HeaderText } from '../Text';
|
|
|
|
import useCookieJar from '../../hooks/useCookieJar';
|
2023-07-20 01:08:23 +00:00
|
|
|
import useIsFirstRender from '../../hooks/useIsFirstRender';
|
2021-07-06 21:52:53 +00:00
|
|
|
|
2022-01-08 00:29:09 +00:00
|
|
|
const PREFIX = 'FullSize';
|
|
|
|
|
|
|
|
const classes = {
|
|
|
|
displayBox: `${PREFIX}-displayBox`,
|
|
|
|
spinnerBox: `${PREFIX}-spinnerBox`,
|
|
|
|
};
|
2021-07-29 21:00:44 +00:00
|
|
|
|
2022-01-08 00:29:09 +00:00
|
|
|
const StyledDiv = styled('div')(() => ({
|
|
|
|
[`& .${classes.displayBox}`]: {
|
2021-08-05 15:12:59 +00:00
|
|
|
width: '75vw',
|
|
|
|
height: '75vh',
|
2021-07-19 22:34:57 +00:00
|
|
|
},
|
2022-01-08 00:29:09 +00:00
|
|
|
|
|
|
|
[`& .${classes.spinnerBox}`]: {
|
2021-07-20 19:53:50 +00:00
|
|
|
flexDirection: 'column',
|
|
|
|
width: '75vw',
|
|
|
|
height: '75vh',
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center',
|
2021-07-06 21:52:53 +00:00
|
|
|
},
|
|
|
|
}));
|
|
|
|
|
2022-01-08 00:29:09 +00:00
|
|
|
const VncDisplay = dynamic(() => import('./VncDisplay'), { ssr: false });
|
|
|
|
|
2023-07-20 05:20:23 +00:00
|
|
|
// Unit: seconds
|
2024-05-22 03:16:00 +00:00
|
|
|
const DEFAULT_VNC_RECONNECT_TIMER_START = 10;
|
2022-06-24 22:58:11 +00:00
|
|
|
|
2024-05-22 05:42:53 +00:00
|
|
|
const MAP_TO_WSCODE_MSG: Record<number, string> = {
|
|
|
|
1000: 'in-use by another process?',
|
|
|
|
1006: 'destination is down?',
|
|
|
|
};
|
|
|
|
|
2023-07-20 19:48:13 +00:00
|
|
|
const buildServerVncUrl = (host: string, serverUuid: string) =>
|
|
|
|
`ws://${host}/ws/server/vnc/${serverUuid}`;
|
2023-07-20 01:08:23 +00:00
|
|
|
|
2022-06-24 22:58:11 +00:00
|
|
|
const FullSize: FC<FullSizeProps> = ({
|
|
|
|
onClickCloseButton,
|
2022-06-24 19:42:19 +00:00
|
|
|
serverUUID,
|
2021-08-10 16:27:33 +00:00
|
|
|
serverName,
|
2023-07-20 05:20:23 +00:00
|
|
|
vncReconnectTimerStart = DEFAULT_VNC_RECONNECT_TIMER_START,
|
2022-06-24 22:58:11 +00:00
|
|
|
}): JSX.Element => {
|
2024-05-22 03:16:00 +00:00
|
|
|
const { buildCookieJar } = useCookieJar();
|
2023-07-20 01:08:23 +00:00
|
|
|
const isFirstRender = useIsFirstRender();
|
2021-05-19 22:23:55 +00:00
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const [anchorEl, setAnchorEl] = useState<HTMLElement | null>(null);
|
2023-07-20 05:20:23 +00:00
|
|
|
const [rfbConnectArgs, setRfbConnectArgs] = useState<
|
|
|
|
Partial<RfbConnectArgs> | undefined
|
2023-07-20 01:08:23 +00:00
|
|
|
>(undefined);
|
2023-07-20 05:20:23 +00:00
|
|
|
const [vncConnecting, setVncConnecting] = useState<boolean>(false);
|
2024-05-22 05:24:25 +00:00
|
|
|
|
2023-07-20 05:20:23 +00:00
|
|
|
const [vncError, setVncError] = useState<boolean>(false);
|
2024-05-22 05:24:25 +00:00
|
|
|
const [vncWsErrorMessage, setVncWsErrorMessage] = useState<
|
2024-05-22 03:16:00 +00:00
|
|
|
string | undefined
|
|
|
|
>();
|
|
|
|
const [vncApiErrorMessage, setVncApiErrorMessage] = useState<
|
|
|
|
string | undefined
|
|
|
|
>();
|
2024-05-22 05:24:25 +00:00
|
|
|
|
2023-07-20 05:20:23 +00:00
|
|
|
const [vncReconnectTimer, setVncReconnectTimer] = useState<number>(
|
|
|
|
vncReconnectTimerStart,
|
|
|
|
);
|
2023-06-16 05:39:39 +00:00
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const rfb = useRef<typeof RFB | null>(null);
|
|
|
|
const rfbScreen = useRef<HTMLDivElement | null>(null);
|
2023-06-16 05:39:39 +00:00
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const handleClickKeyboard = (
|
|
|
|
event: React.MouseEvent<HTMLButtonElement>,
|
|
|
|
): void => {
|
2021-07-12 23:45:21 +00:00
|
|
|
setAnchorEl(event.currentTarget);
|
|
|
|
};
|
|
|
|
|
2021-07-14 15:32:29 +00:00
|
|
|
const handleSendKeys = (scans: string[]) => {
|
2021-07-13 23:58:07 +00:00
|
|
|
if (rfb.current) {
|
2021-07-14 15:32:29 +00:00
|
|
|
if (!scans.length) rfb.current.sendCtrlAltDel();
|
|
|
|
else {
|
|
|
|
// Send pressing keys
|
2021-07-29 21:00:44 +00:00
|
|
|
for (let i = 0; i <= scans.length - 1; i += 1) {
|
|
|
|
rfb.current.sendKey(scans[i], 1);
|
|
|
|
}
|
2021-07-14 15:32:29 +00:00
|
|
|
|
|
|
|
// Send releasing keys in reverse order
|
|
|
|
for (let i = scans.length - 1; i >= 0; i -= 1) {
|
|
|
|
rfb.current.sendKey(scans[i], 0);
|
|
|
|
}
|
|
|
|
}
|
2021-07-13 23:58:07 +00:00
|
|
|
setAnchorEl(null);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const connectServerVnc = useCallback(() => {
|
|
|
|
setVncConnecting(true);
|
|
|
|
setVncError(false);
|
|
|
|
|
|
|
|
setRfbConnectArgs({
|
2023-07-20 19:48:13 +00:00
|
|
|
url: buildServerVncUrl(window.location.host, serverUUID),
|
2023-07-20 01:08:23 +00:00
|
|
|
});
|
2023-07-20 05:20:23 +00:00
|
|
|
}, [serverUUID]);
|
2023-07-20 01:08:23 +00:00
|
|
|
|
|
|
|
const disconnectServerVnc = useCallback(() => {
|
2023-12-01 20:57:21 +00:00
|
|
|
if (rfb?.current) {
|
|
|
|
rfb.current.disconnect();
|
|
|
|
rfb.current = null;
|
|
|
|
}
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
setRfbConnectArgs(undefined);
|
2023-07-20 05:20:23 +00:00
|
|
|
}, []);
|
2023-07-20 01:08:23 +00:00
|
|
|
|
|
|
|
const reconnectServerVnc = useCallback(() => {
|
2023-12-01 20:57:21 +00:00
|
|
|
disconnectServerVnc();
|
2023-07-20 01:08:23 +00:00
|
|
|
connectServerVnc();
|
2023-12-01 20:57:21 +00:00
|
|
|
}, [connectServerVnc, disconnectServerVnc]);
|
2023-07-20 01:08:23 +00:00
|
|
|
|
2023-07-20 05:20:23 +00:00
|
|
|
const updateVncReconnectTimer = useCallback((): void => {
|
|
|
|
const intervalId = setInterval((): void => {
|
|
|
|
setVncReconnectTimer((previous) => {
|
|
|
|
const current = previous - 1;
|
|
|
|
|
|
|
|
if (current < 1) {
|
|
|
|
clearInterval(intervalId);
|
|
|
|
}
|
|
|
|
|
|
|
|
return current;
|
|
|
|
});
|
|
|
|
}, 1000);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
// 'connect' event emits when a connection successfully completes.
|
|
|
|
const rfbConnectEventHandler = useCallback(() => {
|
|
|
|
setVncConnecting(false);
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
// 'disconnect' event emits when a connection fails,
|
|
|
|
// OR when a user closes the existing connection.
|
|
|
|
const rfbDisconnectEventHandler = useCallback(
|
2024-05-22 03:16:00 +00:00
|
|
|
(event) => {
|
|
|
|
const { detail } = event;
|
|
|
|
const { clean } = detail;
|
2023-07-20 05:20:23 +00:00
|
|
|
|
2024-05-22 03:16:00 +00:00
|
|
|
if (clean) return;
|
|
|
|
|
|
|
|
setVncConnecting(false);
|
|
|
|
setVncError(true);
|
|
|
|
|
|
|
|
updateVncReconnectTimer();
|
2023-07-20 05:20:23 +00:00
|
|
|
},
|
|
|
|
[updateVncReconnectTimer],
|
|
|
|
);
|
|
|
|
|
2024-05-22 03:16:00 +00:00
|
|
|
const wsCloseEventHandler = useCallback(
|
|
|
|
(event?: WebsockCloseEvent): void => {
|
2024-05-22 05:24:25 +00:00
|
|
|
if (!event) {
|
|
|
|
setVncWsErrorMessage(undefined);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2024-05-22 03:16:00 +00:00
|
|
|
|
|
|
|
const { code: wscode, reason } = event;
|
|
|
|
|
2024-05-22 05:42:53 +00:00
|
|
|
let wsmsg = `ws: ${wscode}`;
|
|
|
|
|
|
|
|
const guess = MAP_TO_WSCODE_MSG[wscode];
|
|
|
|
|
|
|
|
if (guess) {
|
|
|
|
wsmsg += ` (${guess})`;
|
|
|
|
}
|
2024-05-22 03:16:00 +00:00
|
|
|
|
|
|
|
if (reason) {
|
2024-05-22 05:42:53 +00:00
|
|
|
wsmsg += `, ${reason}`;
|
2024-05-22 03:16:00 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 05:42:53 +00:00
|
|
|
setVncWsErrorMessage(wsmsg);
|
2024-05-22 03:16:00 +00:00
|
|
|
|
|
|
|
const vncerror = buildCookieJar()[
|
|
|
|
`suiapi.vncerror.${serverUUID}`
|
|
|
|
] as APIError;
|
|
|
|
|
2024-05-22 05:24:25 +00:00
|
|
|
if (!vncerror) {
|
|
|
|
setVncApiErrorMessage(undefined);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2024-05-22 03:16:00 +00:00
|
|
|
|
|
|
|
const { code: apicode, message } = vncerror;
|
|
|
|
|
|
|
|
setVncApiErrorMessage(`api: ${apicode}, ${message}`);
|
|
|
|
},
|
|
|
|
[buildCookieJar, serverUUID],
|
|
|
|
);
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const showScreen = useMemo(
|
|
|
|
() => !vncConnecting && !vncError,
|
|
|
|
[vncConnecting, vncError],
|
|
|
|
);
|
|
|
|
|
2024-03-23 03:37:39 +00:00
|
|
|
const fullscreenElement = useMemo(
|
|
|
|
() => (
|
|
|
|
<Box>
|
|
|
|
<IconButton
|
|
|
|
onClick={() => {
|
|
|
|
rfbScreen.current?.requestFullscreen();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<FullscreenIcon />
|
|
|
|
</IconButton>
|
|
|
|
</Box>
|
|
|
|
),
|
|
|
|
[],
|
|
|
|
);
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const keyboardMenuElement = useMemo(
|
|
|
|
() => (
|
|
|
|
<Box>
|
|
|
|
<IconButton onClick={handleClickKeyboard}>
|
|
|
|
<KeyboardIcon />
|
|
|
|
</IconButton>
|
|
|
|
<Menu
|
|
|
|
anchorEl={anchorEl}
|
|
|
|
keepMounted
|
|
|
|
open={Boolean(anchorEl)}
|
|
|
|
onClose={() => setAnchorEl(null)}
|
|
|
|
>
|
|
|
|
{keyCombinations.map(({ keys, scans }) => (
|
2023-07-20 05:33:58 +00:00
|
|
|
<MenuItem key={keys} onClick={() => handleSendKeys(scans)}>
|
2023-07-20 01:08:23 +00:00
|
|
|
<Typography variant="subtitle1">{keys}</Typography>
|
|
|
|
</MenuItem>
|
|
|
|
))}
|
|
|
|
</Menu>
|
|
|
|
</Box>
|
|
|
|
),
|
|
|
|
[anchorEl],
|
|
|
|
);
|
|
|
|
|
|
|
|
const vncDisconnectElement = useMemo(
|
|
|
|
() => (
|
2023-12-01 21:03:05 +00:00
|
|
|
<Box>
|
|
|
|
<IconButton
|
|
|
|
onClick={(...args) => {
|
|
|
|
disconnectServerVnc();
|
|
|
|
onClickCloseButton?.call(null, ...args);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<CloseIcon />
|
|
|
|
</IconButton>
|
|
|
|
</Box>
|
2023-07-20 01:08:23 +00:00
|
|
|
),
|
|
|
|
[disconnectServerVnc, onClickCloseButton],
|
|
|
|
);
|
|
|
|
|
2023-12-01 20:57:21 +00:00
|
|
|
const returnHomeElement = useMemo(
|
|
|
|
() => (
|
2023-12-01 21:03:05 +00:00
|
|
|
<Box>
|
|
|
|
<IconButton
|
|
|
|
onClick={() => {
|
|
|
|
if (!window) return;
|
2023-12-01 20:57:21 +00:00
|
|
|
|
2023-12-01 21:03:05 +00:00
|
|
|
disconnectServerVnc();
|
2023-12-01 20:57:21 +00:00
|
|
|
|
2023-12-01 21:03:05 +00:00
|
|
|
window.location.assign('/');
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<DashboardIcon />
|
|
|
|
</IconButton>
|
|
|
|
</Box>
|
2023-12-01 20:57:21 +00:00
|
|
|
),
|
|
|
|
[disconnectServerVnc],
|
|
|
|
);
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
const vncToolbarElement = useMemo(
|
|
|
|
() =>
|
|
|
|
showScreen && (
|
|
|
|
<>
|
2024-03-23 03:37:39 +00:00
|
|
|
{fullscreenElement}
|
2023-07-20 01:08:23 +00:00
|
|
|
{keyboardMenuElement}
|
2023-12-13 20:38:31 +00:00
|
|
|
<ServerMenu
|
|
|
|
serverName={serverName}
|
|
|
|
serverState="running"
|
|
|
|
serverUuid={serverUUID}
|
|
|
|
/>
|
2023-12-01 20:57:21 +00:00
|
|
|
{returnHomeElement}
|
2023-07-20 01:08:23 +00:00
|
|
|
{vncDisconnectElement}
|
|
|
|
</>
|
|
|
|
),
|
2023-12-13 20:38:31 +00:00
|
|
|
[
|
2024-03-23 03:37:39 +00:00
|
|
|
fullscreenElement,
|
2023-12-13 20:38:31 +00:00
|
|
|
keyboardMenuElement,
|
|
|
|
returnHomeElement,
|
|
|
|
serverName,
|
|
|
|
serverUUID,
|
|
|
|
showScreen,
|
|
|
|
vncDisconnectElement,
|
|
|
|
],
|
2023-07-20 01:08:23 +00:00
|
|
|
);
|
|
|
|
|
2023-07-20 05:20:23 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (vncReconnectTimer === 0) {
|
|
|
|
setVncReconnectTimer(vncReconnectTimerStart);
|
|
|
|
|
|
|
|
reconnectServerVnc();
|
|
|
|
}
|
|
|
|
}, [reconnectServerVnc, vncReconnectTimer, vncReconnectTimerStart]);
|
|
|
|
|
2023-07-20 01:08:23 +00:00
|
|
|
useEffect(() => {
|
|
|
|
if (isFirstRender) {
|
|
|
|
connectServerVnc();
|
|
|
|
}
|
|
|
|
}, [connectServerVnc, isFirstRender]);
|
|
|
|
|
2021-05-19 22:23:55 +00:00
|
|
|
return (
|
|
|
|
<Panel>
|
2023-07-20 01:08:23 +00:00
|
|
|
<PanelHeader>
|
|
|
|
<HeaderText text={`Server: ${serverName}`} />
|
|
|
|
{vncToolbarElement}
|
|
|
|
</PanelHeader>
|
2022-01-08 00:29:09 +00:00
|
|
|
<StyledDiv>
|
2023-07-20 01:08:23 +00:00
|
|
|
<Box
|
|
|
|
display={showScreen ? 'flex' : 'none'}
|
|
|
|
className={classes.displayBox}
|
|
|
|
>
|
|
|
|
<VncDisplay
|
2023-07-20 05:20:23 +00:00
|
|
|
onConnect={rfbConnectEventHandler}
|
|
|
|
onDisconnect={rfbDisconnectEventHandler}
|
2024-05-22 03:16:00 +00:00
|
|
|
onWsClose={wsCloseEventHandler}
|
2023-07-20 01:08:23 +00:00
|
|
|
rfb={rfb}
|
2023-07-20 05:20:23 +00:00
|
|
|
rfbConnectArgs={rfbConnectArgs}
|
2023-07-20 01:08:23 +00:00
|
|
|
rfbScreen={rfbScreen}
|
|
|
|
/>
|
2021-08-05 15:12:59 +00:00
|
|
|
</Box>
|
2023-07-20 01:08:23 +00:00
|
|
|
{!showScreen && (
|
2024-05-22 03:16:00 +00:00
|
|
|
<Box display="flex" className={classes.spinnerBox} textAlign="center">
|
2023-07-20 01:08:23 +00:00
|
|
|
{vncConnecting && (
|
2022-01-08 00:29:09 +00:00
|
|
|
<>
|
2024-05-22 03:16:00 +00:00
|
|
|
<HeaderText>Connecting to {serverName}.</HeaderText>
|
2022-01-08 00:29:09 +00:00
|
|
|
<Spinner />
|
|
|
|
</>
|
2023-07-20 01:08:23 +00:00
|
|
|
)}
|
|
|
|
{vncError && (
|
2022-01-08 00:29:09 +00:00
|
|
|
<>
|
2024-05-22 03:16:00 +00:00
|
|
|
<HeaderText>Can't connect to the server.</HeaderText>
|
|
|
|
<BodyText>{vncApiErrorMessage}</BodyText>
|
2024-05-22 05:24:25 +00:00
|
|
|
<BodyText>{vncWsErrorMessage}</BodyText>
|
2024-05-22 03:16:00 +00:00
|
|
|
<HeaderText mt=".5em">
|
2023-07-20 05:20:23 +00:00
|
|
|
Retrying in {vncReconnectTimer}.
|
|
|
|
</HeaderText>
|
2022-01-08 00:29:09 +00:00
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</Box>
|
|
|
|
)}
|
|
|
|
</StyledDiv>
|
2021-05-19 22:23:55 +00:00
|
|
|
</Panel>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2021-07-06 21:52:53 +00:00
|
|
|
export default FullSize;
|