2016-07-30 22:47:08 +00:00
|
|
|
# Offscreen Rendering
|
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
## Overview
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
Offscreen rendering lets you obtain the content of a `BrowserWindow` in a
|
2024-11-07 14:41:09 +00:00
|
|
|
bitmap or a shared GPU texture, so it can be rendered anywhere, for example,
|
|
|
|
on texture in a 3D scene.
|
2020-12-01 01:47:09 +00:00
|
|
|
The offscreen rendering in Electron uses a similar approach to that of the
|
|
|
|
[Chromium Embedded Framework](https://bitbucket.org/chromiumembedded/cef)
|
|
|
|
project.
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2023-04-03 11:20:10 +00:00
|
|
|
_Notes_:
|
2016-12-21 22:15:39 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
* There are two rendering modes that can be used (see the section below) and only
|
|
|
|
the dirty area is passed to the `paint` event to be more efficient.
|
|
|
|
* You can stop/continue the rendering as well as set the frame rate.
|
2020-12-08 04:41:09 +00:00
|
|
|
* The maximum frame rate is 240 because greater values bring only performance
|
2020-12-01 01:47:09 +00:00
|
|
|
losses with no benefits.
|
|
|
|
* When nothing is happening on a webpage, no frames are generated.
|
|
|
|
* An offscreen window is always created as a
|
2024-11-07 14:41:09 +00:00
|
|
|
[Frameless Window](../tutorial/window-customization.md).
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
### Rendering Modes
|
|
|
|
|
|
|
|
#### GPU accelerated
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2024-11-07 14:41:09 +00:00
|
|
|
GPU accelerated rendering means that the GPU is used for composition. The benefit
|
|
|
|
of this mode is that WebGL and 3D CSS animations are supported. There are two
|
|
|
|
different approaches depending on the `webPreferences.offscreen.useSharedTexture`
|
|
|
|
setting.
|
|
|
|
|
|
|
|
1. Use GPU shared texture
|
|
|
|
|
|
|
|
Used when `webPreferences.offscreen.useSharedTexture` is set to `true`.
|
|
|
|
|
|
|
|
This is an advanced feature requiring a native node module to work with your own code.
|
|
|
|
The frames are directly copied in GPU textures, thus this mode is very fast because
|
|
|
|
there's no CPU-GPU memory copies overhead, and you can directly import the shared
|
|
|
|
texture to your own rendering program.
|
|
|
|
|
|
|
|
2. Use CPU shared memory bitmap
|
|
|
|
|
|
|
|
Used when `webPreferences.offscreen.useSharedTexture` is set to `false` (default behavior).
|
|
|
|
|
|
|
|
The texture is accessible using the `NativeImage` API at the cost of performance.
|
|
|
|
The frame has to be copied from the GPU to the CPU bitmap which requires more system
|
|
|
|
resources, thus this mode is slower than the Software output device mode. But it supports
|
|
|
|
GPU related functionalities.
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
#### Software output device
|
2016-07-30 22:47:08 +00:00
|
|
|
|
|
|
|
This mode uses a software output device for rendering in the CPU, so the frame
|
2024-11-07 14:41:09 +00:00
|
|
|
generation is faster than shared memory bitmap GPU accelerated mode.
|
2016-08-03 01:15:38 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
To enable this mode, GPU acceleration has to be disabled by calling the
|
2016-08-03 01:32:29 +00:00
|
|
|
[`app.disableHardwareAcceleration()`][disablehardwareacceleration] API.
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
## Example
|
|
|
|
|
2023-11-21 07:50:08 +00:00
|
|
|
```fiddle docs/fiddles/features/offscreen-rendering
|
|
|
|
const { app, BrowserWindow } = require('electron/main')
|
2023-08-30 15:55:23 +00:00
|
|
|
const fs = require('node:fs')
|
2023-11-21 07:50:08 +00:00
|
|
|
const path = require('node:path')
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2016-08-03 01:32:29 +00:00
|
|
|
app.disableHardwareAcceleration()
|
2016-07-30 22:47:08 +00:00
|
|
|
|
2023-11-21 07:50:08 +00:00
|
|
|
function createWindow () {
|
|
|
|
const win = new BrowserWindow({
|
|
|
|
width: 800,
|
|
|
|
height: 600,
|
|
|
|
webPreferences: {
|
|
|
|
offscreen: true
|
|
|
|
}
|
|
|
|
})
|
2018-02-19 23:50:26 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
win.loadURL('https://github.com')
|
2016-08-04 04:47:52 +00:00
|
|
|
win.webContents.on('paint', (event, dirty, image) => {
|
2020-12-01 01:47:09 +00:00
|
|
|
fs.writeFileSync('ex.png', image.toPNG())
|
2016-08-04 04:47:52 +00:00
|
|
|
})
|
2020-12-01 01:47:09 +00:00
|
|
|
win.webContents.setFrameRate(60)
|
2023-11-21 07:50:08 +00:00
|
|
|
console.log(`The screenshot has been successfully saved to ${path.join(process.cwd(), 'ex.png')}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
app.whenReady().then(() => {
|
|
|
|
createWindow()
|
|
|
|
|
|
|
|
app.on('activate', () => {
|
|
|
|
if (BrowserWindow.getAllWindows().length === 0) {
|
|
|
|
createWindow()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
app.on('window-all-closed', () => {
|
|
|
|
if (process.platform !== 'darwin') {
|
|
|
|
app.quit()
|
|
|
|
}
|
2016-08-03 01:15:38 +00:00
|
|
|
})
|
2016-07-30 22:47:08 +00:00
|
|
|
```
|
2016-08-03 01:32:29 +00:00
|
|
|
|
2020-12-01 01:47:09 +00:00
|
|
|
After launching the Electron application, navigate to your application's
|
2021-05-24 02:33:45 +00:00
|
|
|
working folder, where you'll find the rendered image.
|
2024-05-10 09:00:15 +00:00
|
|
|
|
2016-08-03 01:32:29 +00:00
|
|
|
[disablehardwareacceleration]: ../api/app.md#appdisablehardwareacceleration
|