From aed3ad9694bcb1f112c869d2c2bdf41b1078e340 Mon Sep 17 00:00:00 2001 From: brentyi Date: Sun, 13 Oct 2024 21:11:58 -0700 Subject: [PATCH] Fix temporal off-by-1 for wxyz/position updates --- src/viser/client/src/MessageHandler.tsx | 199 +++++++++++++----------- src/viser/client/src/SceneTree.tsx | 6 +- 2 files changed, 112 insertions(+), 93 deletions(-) diff --git a/src/viser/client/src/MessageHandler.tsx b/src/viser/client/src/MessageHandler.tsx index 09771e44c..f00ea2dcf 100644 --- a/src/viser/client/src/MessageHandler.tsx +++ b/src/viser/client/src/MessageHandler.tsx @@ -522,103 +522,118 @@ export function FrameSynchronizedMessageHandler() { const viewer = useContext(ViewerContext)!; const messageQueueRef = viewer.messageQueueRef; - useFrame(() => { - // Send a render along if it was requested! - if (viewer.getRenderRequestState.current === "triggered") { - viewer.getRenderRequestState.current = "pause"; - } else if (viewer.getRenderRequestState.current === "pause") { - const sourceCanvas = viewer.canvasRef.current!; - - const targetWidth = viewer.getRenderRequest.current!.width; - const targetHeight = viewer.getRenderRequest.current!.height; - - // We'll save a render to an intermediate canvas with the requested dimensions. - const renderBufferCanvas = new OffscreenCanvas(targetWidth, targetHeight); - const ctx = renderBufferCanvas.getContext("2d")!; - ctx.reset(); - // Use a white background for JPEGs, which don't have an alpha channel. - if (viewer.getRenderRequest.current?.format === "image/jpeg") { - ctx.fillStyle = "white"; - ctx.fillRect(0, 0, renderBufferCanvas.width, renderBufferCanvas.height); - } - - // Determine offsets for the source canvas. We'll always center our renders. - // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/drawImage - let sourceWidth = sourceCanvas.width; - let sourceHeight = sourceCanvas.height; - - const sourceAspect = sourceWidth / sourceHeight; - const targetAspect = targetWidth / targetHeight; - - if (sourceAspect > targetAspect) { - // The source is wider than the target. - // We need to shrink the width. - sourceWidth = Math.round(targetAspect * sourceHeight); - } else if (sourceAspect < targetAspect) { - // The source is narrower than the target. - // We need to shrink the height. - sourceHeight = Math.round(sourceWidth / targetAspect); - } - - console.log( - `Sending render; requested aspect ratio was ${targetAspect} (dimensinos: ${targetWidth}/${targetHeight}), copying from aspect ratio ${ - sourceWidth / sourceHeight - } (dimensions: ${sourceWidth}/${sourceHeight}).`, - ); + useFrame( + () => { + // Send a render along if it was requested! + if (viewer.getRenderRequestState.current === "triggered") { + viewer.getRenderRequestState.current = "pause"; + } else if (viewer.getRenderRequestState.current === "pause") { + const sourceCanvas = viewer.canvasRef.current!; + + const targetWidth = viewer.getRenderRequest.current!.width; + const targetHeight = viewer.getRenderRequest.current!.height; + + // We'll save a render to an intermediate canvas with the requested dimensions. + const renderBufferCanvas = new OffscreenCanvas( + targetWidth, + targetHeight, + ); + const ctx = renderBufferCanvas.getContext("2d")!; + ctx.reset(); + // Use a white background for JPEGs, which don't have an alpha channel. + if (viewer.getRenderRequest.current?.format === "image/jpeg") { + ctx.fillStyle = "white"; + ctx.fillRect( + 0, + 0, + renderBufferCanvas.width, + renderBufferCanvas.height, + ); + } - ctx.drawImage( - sourceCanvas, - (sourceCanvas.width - sourceWidth) / 2.0, - (sourceCanvas.height - sourceHeight) / 2.0, - sourceWidth, - sourceHeight, - 0, - 0, - targetWidth, - targetHeight, - ); + // Determine offsets for the source canvas. We'll always center our renders. + // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D/drawImage + let sourceWidth = sourceCanvas.width; + let sourceHeight = sourceCanvas.height; + + const sourceAspect = sourceWidth / sourceHeight; + const targetAspect = targetWidth / targetHeight; + + if (sourceAspect > targetAspect) { + // The source is wider than the target. + // We need to shrink the width. + sourceWidth = Math.round(targetAspect * sourceHeight); + } else if (sourceAspect < targetAspect) { + // The source is narrower than the target. + // We need to shrink the height. + sourceHeight = Math.round(sourceWidth / targetAspect); + } + + console.log( + `Sending render; requested aspect ratio was ${targetAspect} (dimensinos: ${targetWidth}/${targetHeight}), copying from aspect ratio ${ + sourceWidth / sourceHeight + } (dimensions: ${sourceWidth}/${sourceHeight}).`, + ); - viewer.getRenderRequestState.current = "in_progress"; - - // Encode the image, the send it. - renderBufferCanvas - .convertToBlob({ - type: viewer.getRenderRequest.current!.format, - quality: viewer.getRenderRequest.current!.quality / 100.0, - }) - .then(async (blob) => { - if (blob === null) { - console.error("Render failed"); + ctx.drawImage( + sourceCanvas, + (sourceCanvas.width - sourceWidth) / 2.0, + (sourceCanvas.height - sourceHeight) / 2.0, + sourceWidth, + sourceHeight, + 0, + 0, + targetWidth, + targetHeight, + ); + + viewer.getRenderRequestState.current = "in_progress"; + + // Encode the image, the send it. + renderBufferCanvas + .convertToBlob({ + type: viewer.getRenderRequest.current!.format, + quality: viewer.getRenderRequest.current!.quality / 100.0, + }) + .then(async (blob) => { + if (blob === null) { + console.error("Render failed"); + viewer.getRenderRequestState.current = "ready"; + return; + } + const payload = new Uint8Array(await blob.arrayBuffer()); + viewer.sendMessageRef.current({ + type: "GetRenderResponseMessage", + payload: payload, + }); viewer.getRenderRequestState.current = "ready"; - return; - } - const payload = new Uint8Array(await blob.arrayBuffer()); - viewer.sendMessageRef.current({ - type: "GetRenderResponseMessage", - payload: payload, }); - viewer.getRenderRequestState.current = "ready"; - }); - } + } - // Handle messages, but only if we're not trying to render something. - if (viewer.getRenderRequestState.current === "ready") { - // Handle messages before every frame. - // Place this directly in ws.onmessage can cause race conditions! - // - // If a render is requested, note that we don't handle any more messages - // until the render is done. - const requestRenderIndex = messageQueueRef.current.findIndex( - (message) => message.type === "GetRenderRequestMessage", - ); - const numMessages = - requestRenderIndex !== -1 - ? requestRenderIndex + 1 - : messageQueueRef.current.length; - const processBatch = messageQueueRef.current.splice(0, numMessages); - processBatch.forEach(handleMessage); - } - }); + // Handle messages, but only if we're not trying to render something. + if (viewer.getRenderRequestState.current === "ready") { + // Handle messages before every frame. + // Place this directly in ws.onmessage can cause race conditions! + // + // If a render is requested, note that we don't handle any more messages + // until the render is done. + const requestRenderIndex = messageQueueRef.current.findIndex( + (message) => message.type === "GetRenderRequestMessage", + ); + const numMessages = + requestRenderIndex !== -1 + ? requestRenderIndex + 1 + : messageQueueRef.current.length; + const processBatch = messageQueueRef.current.splice(0, numMessages); + processBatch.forEach(handleMessage); + } + }, + // We should handle messages before doing anything else!! + // + // Importantly, this priority should be *lower* than the useFrame priority + // used to update scene node transforms in SceneTree.tsx. + -100000, + ); return null; } diff --git a/src/viser/client/src/SceneTree.tsx b/src/viser/client/src/SceneTree.tsx index 51adcf534..1904bab56 100644 --- a/src/viser/client/src/SceneTree.tsx +++ b/src/viser/client/src/SceneTree.tsx @@ -704,7 +704,11 @@ export function SceneNodeThreeObject(props: { }, // Other useFrame hooks may depend on transforms + visibility. So it's best // to call this hook early. - -10000, + // + // However, it's also important that this is *higher* than the priority for + // the MessageHandler's useFrame. This is to make sure that transforms are + // updated in the same frame that they are set. + -1000, ); // Clicking logic.