anvil/striker-ui/components/Display/VncDisplay.tsx

118 lines
2.5 KiB
TypeScript

import RFB from '@novnc/novnc/core/rfb';
import { useEffect } from 'react';
const rfbConnect: RfbConnectFunction = ({
background = '',
clipViewport = false,
compressionLevel = 2,
dragViewport = false,
focusOnClick = false,
onConnect,
onDisconnect,
qualityLevel = 6,
resizeSession = true,
rfb,
rfbScreen,
scaleViewport = true,
showDotCursor = false,
url,
viewOnly = false,
}) => {
if (!rfbScreen?.current || rfb?.current) return;
rfbScreen.current.innerHTML = '';
rfb.current = new RFB(rfbScreen.current, url);
rfb.current.background = background;
rfb.current.clipViewport = clipViewport;
rfb.current.compressionLevel = compressionLevel;
rfb.current.dragViewport = dragViewport;
rfb.current.focusOnClick = focusOnClick;
rfb.current.qualityLevel = qualityLevel;
rfb.current.resizeSession = resizeSession;
rfb.current.scaleViewport = scaleViewport;
rfb.current.showDotCursor = showDotCursor;
rfb.current.viewOnly = viewOnly;
// RFB extends custom class EventTargetMixin;
// the usual .on or .once doesn't exist.
if (onConnect) {
rfb.current.addEventListener('connect', onConnect);
}
if (onDisconnect) {
rfb.current.addEventListener('disconnect', onDisconnect);
}
};
const rfbDisconnect: RfbDisconnectFunction = (rfb) => {
if (!rfb?.current) return;
rfb.current.disconnect();
rfb.current = null;
};
const VncDisplay = (props: VncDisplayProps): JSX.Element => {
const {
onConnect,
onDisconnect,
rfb,
rfbConnectArgs,
rfbScreen,
url: initUrl,
} = props;
useEffect(() => {
if (rfbConnectArgs) {
const { url = initUrl } = rfbConnectArgs;
if (!url) return;
const args: RfbConnectArgs = {
onConnect,
onDisconnect,
rfb,
rfbScreen,
url,
...rfbConnectArgs,
};
rfbConnect(args);
} else {
rfbDisconnect(rfb);
}
}, [initUrl, onConnect, onDisconnect, rfb, rfbConnectArgs, rfbScreen]);
useEffect(
() => () => {
rfbDisconnect(rfb);
},
[rfb],
);
const handleMouseEnter = () => {
if (
document.activeElement &&
document.activeElement instanceof HTMLElement
) {
document.activeElement.blur();
}
if (rfb?.current) rfb.current.focus();
};
return (
<div
style={{ width: '100%', height: '75vh' }}
ref={rfbScreen}
onMouseEnter={handleMouseEnter}
/>
);
};
VncDisplay.displayName = 'VncDisplay';
export default VncDisplay;