A React hook for automatically exporting a canvas
to img
.
- For session recording services like FullStory and LogRocket that are unable to record the contents of a
canvas
element, this provides a workaround. By default, theimg
is rendered directly behind the canvas.- A positioned parent element should wrap the canvas to ensure the
img
renders in the correct position. - If you have UI rendering above the canvas, you may need to make
z-index
tweaks to ensure the canvas stays behind the UI. By default, this hook sets theimg
toz-index: 0
and thecanvas
toz-index: 1
.
- A positioned parent element should wrap the canvas to ensure the
- Can provide a downloadable image of the current state of a
canvas
. Useful for drawing apps, meme generators, etc.
Note: FullStory adds a semi-transparent striped overlay in place of elements with .fs-exclude
. While you can still see the img
in session replays, it’s rendered with those stripes over it. We’re working with FullStory to provide a way to completely hide the element.
When using a canvas with a WebGL context that you don’t have direct access to manipulate (like three.js):
- Set the canvas to use
preserveDrawingBuffer: true
. (This can be passed toTHREE.WebGLRenderer
if using three.js.) - Use the
interval
property to define how often theimg
is automatically updated.
Because we’re not able to insert our function directly into the drawing thread when using 3rd party apps that provide no such callback, we have to poll. This requires the drawing buffer to be preserved so we can read it in a later thread. This may result in a small performance hit due to the context switching from swap to copy. Depending on your use case, this may or may not be an issue.
Example use with FullStory’s .fs-exclude
class.
// in component that renders canvas
const canvasRef = useRef()
const updateImage = useCanvasImage({
canvas: canvasRef,
className: 'fs-exclude', // to hide from FullStory
})
useEffect(() => {
// in this simple example, a change to someVar indicates the canvas has been updated
updateImage()
}, [someVar])
// JSX
<div style="position: relative">
<canvas ref={canvasRef} />
</div>
/* html output */
<div style="position: relative">
<canvas class="fs-exclude" />
<img />
</div>
Example use with @react-three/xr.
useCanvasImage({
canvas: '[data-engine~="three.js"]',
className: 'fs-exclude',
interval: 100,
})
// JSX, using VRCanvas from @react-three/xr
<div style="position: relative">
<VRCanvas
// passed to THREE.WebGLRenderer and then to the canvas context
gl={{ preserveDrawingBuffer: true }}
/>
</div>
/* html output */
<div style="position: relative">
<canvas class="fs-exclude" />
<img />
</div>
canvas
(React ref, string selector, or pre-selected HTML canvas element). If string, must be unique in dom and formatted as selector (#id, .class, [attr="val"], etc).imgClassname
(string) for styling output image, space separated.canvasClassName
(string) for adding.fs-exclude
(FullStory) or other class names to theimg
, space separated.canvasAttributes
(object) for addingdata-private
(LogRocket) or other attributes to thecanvas
.fileType
(string, image/jpeg, image/png, image/webp) what img format to output. default, recommended for session recording: image/jpegquality
(number, 0-1). compression quality. only applies to image/jpeg or image/webp. default: 0.5interval
(number or null/false, millisecond). how often to update the image. set to null to not update on an interval and to use the explicit function instead. when set to number, returned function becomes no-op and polling is used. default: null- Returns: function to be called when the image should be updated (usually when the canvas is drawn to). no-op when an
interval
is specified.
Please do.