Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/vinlic/webvideocreator

๐ŸŒˆ A framework for rendering web animations into videos. It's implemented based on Node.js + Puppeteer + Chrome + FFmpeg, utilizing the latest browser APIs.
https://github.com/vinlic/webvideocreator

animation chrome ffmpeg nodejs puppeteer video vtuber

Last synced: about 3 hours ago
JSON representation

๐ŸŒˆ A framework for rendering web animations into videos. It's implemented based on Node.js + Puppeteer + Chrome + FFmpeg, utilizing the latest browser APIs.

Awesome Lists containing this project

README

        

[็ฎ€ไฝ“ไธญๆ–‡](./README.md) | [English](./README.en-US.md)



[![NPM](https://nodei.co/npm/web-video-creator.png)](https://npmjs.org/package/web-video-creator)

![GitHub](https://img.shields.io/github/license/Vinlic/WebVideoCreator)
![GitHub package.json version](https://img.shields.io/github/package-json/v/Vinlic/WebVideoCreator)
![npm](https://img.shields.io/npm/dt/web-video-creator)
![GitHub Repo stars](https://img.shields.io/github/stars/Vinlic/WebVideoCreator)

# Introduction

๐ŸŒˆ WebVideoCreator (abbreviated as WVC) is a framework for rendering web animations into videos. It's implemented based on Node.js + Puppeteer + Chrome + FFmpeg. It performs deterministic rendering and captures any HTML5-playable animations (CSS3 animations/SVG animations/Lottie animations/GIF animations/APNG animations/WEBP animations) and any timeline-based animations driven by [RAF](https://developer.mozilla.org/en-US/docs/Web/API/Window/requestAnimationFrame). You can also mischievously use `setInterval` or `setTimeout` to control animations. WVC supports embedding or exporting MP4 and transparent WebM videos, as well as features such as transition compositing, audio synthesis, and font loading. Let's get started with the [Quick Start](#quick-start) ๐Ÿป.

WVC creates a virtual time environment๐Ÿ•’ for your cool animated pages, which can be imagined as something akin to "The Truman Show." Its main responsibility is to transform an [uncertain rendering environment](./docs/renderer-env.md#ไธ็กฎๅฎšๆ€ง็š„ๆธฒๆŸ“็Žฏๅขƒ) into a [deterministic rendering environment](./docs/renderer-env.md#็กฎๅฎšๆ€ง็š„ๆธฒๆŸ“็Žฏๅขƒ).

All of this is made possible by Chrome's provided [deterministic rendering mode](https://goo.gle/chrome-headless-rendering) and support for the headless experimental API: [HeadlessExperimental.beginFrame](https://chromedevtools.github.io/devtools-protocol/tot/HeadlessExperimental/#method-beginFrame), This is an innovative experimental function ๐Ÿงช.

Q&A and discussion group on QQ๐Ÿง: 752693580


# Features

- Developed using Node.js, it's very easy to use, extend, and develop further.
- Video processing is incredibly fast, rendering a video as long as 5 minutes can be completed in just 1 minute, Check [Performance Tips](#Performance-Tips) for optimal performance.
- Supports rendering and compositing of single scenes and multi-scene videos, with the ability to apply transition effects to multi-scene videos.
- Supports chunked video compositing, allowing chunks to be distributed to multiple devices for rendering, then combined into multi-scene videos, significantly reducing rendering time for long videos.
- Supports parallel rendering and compositing of multiple video tasks.
- Support embedding or exporting webm format videos that support transparent channels.
- API support for [distributed rendering](#distributed-rendering-solution), enabling the distribution of a large number of videos to multiple devices for rendering and final merging output with minimal wrapping of WVC.
- Supports GPU acceleration for rendering and compositing, reducing video rendering time significantly.
- Can be deployed and run on both Windows and Linux platforms, On MacOS, it is necessary to enable [Compatible Rendering Mode](#compatible-rendering-mode).


# What is it for?

WVC achieves perfect frame-by-frame capture of any animation on a web page. It minimizes the cost of achieving a WYSIWYG (What You See Is What You Get) experience. As a rendering backend, it currently explores the following applications:

**๐Ÿ“Š Data Visualization Video Rendering**: Combine chart libraries such as ECharts to create animated charts on the web and capture them as videos using WVC. This is useful for creating dynamic ranking videos commonly seen on platforms like TikTok, especially when combined with web scraping for data collection.

**๐Ÿ‘ฉโ€๐Ÿซ Digital Human Video Rendering**: With the popularity of AI-generated characters, various digital avatars appear on screens. WVC supports the use of transparent channel videos or mask videos on web pages, enhancing visual effects by incorporating digital humans into animations.

**๐ŸŽจ Content Creation Video Rendering**: Design a simple frontend animation editor and previewer to meet content creation needs. Use WVC as the backend to achieve a WYSIWYG video effect.

**๐ŸŽฎ๏ธ Game or User Operation Replay Video Rendering**: Integrate with web-based games or applications to capture replays as videos, providing users with easy sharing and secondary editing options.

There are more application scenarios waiting for your exploration, and if you have great ideas, remember to raise an issue ๐Ÿ™‹โ€โ™€๏ธ...

## Advantages over Screen Recording Tools?

**๐Ÿ’ฏ Perfect Capture**: The browser's frame composer defaults to a throttling strategy to reduce resource consumption. When drawing complex graphics or under increased system load, web animations may experience frame skipping, dropping frames, or delays. Using screen recording tools makes it difficult to ensure that every frame is captured correctly. In contrast, WVC takes control of the timing, determining when to draw the next frame onto the screen.

**๐ŸŽž๏ธ Parallel Rendering**: Screen recording tools typically cannot capture the animation content of multiple tab pages simultaneously. However, WVC can capture animations in parallel across multiple pages and ultimately combine these chunks into a single long video, with support for seamless transitions.

**๐Ÿฆพ Automation-Friendly**: Screen recording tools require manual operation. With WVC, a set of web animation templates can be combined with data scrapers and scheduled tasks to automate video production.

**๐Ÿงฉ Quick Integration**: Screen recording tools are challenging to integrate. WVC, developed as an NPM package based on Node.js, can be quickly integrated on the backend. Some developers use it to capture game replays as videos.


# Video Demos

We're also in need of animators! If you're passionate about open source projects, feel free to join us ๐Ÿ˜†.

Check out all the demos here: **[Rendering Example Page](https://github.com/Vinlic/WebVideoCreator/wiki/Rendering-Example)** ๐Ÿค—


# Supported Animation Libraries

In theory, all web animation/graphics libraries should work smoothly in the WVC environment. Below, I've listed only the libraries that I have verified to be compatible:

[Anime.js](https://animejs.com/) / [GSAP](https://greensock.com/) / [D3.js](https://d3js.org/) / [Three.js](https://threejs.org/) / [Echart](https://echarts.apache.org/) / [Lottie-Web](http://airbnb.io/lottie/#/web) / [PixiJS](https://pixijs.download/release/docs/index.html) / [Animate.css](https://animate.style/) / [Mo.js](https://mojs.github.io/) / [Tween.js](https://tweenjs.github.io/tween.js/)

Please note that if you manually use [RAF](https://developer.mozilla.org/en-US/docs/Web/API/Window/requestAnimationFrame) to drive animations, ensure that you receive the `timestamp` parameter from the callback to set the animation's progress to that timestamp. Otherwise, frame rate asynchrony may occur.


# Quick Start

## Installation

```shell
# Install WebVideoCreator from NPM
npm i web-video-creator
```

If you encounter issues with the download of `ffmpeg-static`, please set the environment variable: `FFMPEG_BINARIES_URL=https://cdn.npmmirror.com/binaries/ffmpeg-static`.

## Create a Local Server

WVC needs to capture animations from web pages, and you can create a temporary web server locally to serve static pages for testing purposes. One of the simplest ways to do this is by using `live-server`. If you already have static web pages, you can skip this step.

```shell
# Install live-server globally from NPM
npm i -g live-server
# Start the web server
live-server
```

Create a test page in the root directory of the web server. The following HTML content demonstrates an automatically rotating red triangle SVG animation:

```html



Test Page







```

## Rendering Single Video

```javascript
import WebVideoCreator, { VIDEO_ENCODER, logger } from "web-video-creator";

const wvc = new WebVideoCreator();

// Configure WVC
wvc.config({
// Choose an appropriate encoder based on your hardware. Here, we are using the h264_nvenc encoder for Nvidia graphics cards.
// Encoder selection can refer to docs/video-encoder.md
mp4Encoder: VIDEO_ENCODER.NVIDIA.H264
});

// Create a single-scene video
const video = wvc.createSingleVideo({
// URL of the page to be rendered
url: "http://localhost:8080/test.html",
// or can directly set the page content:
// content: "

Hello WebVideoCreator

",
// Video width
width: 1280,
// Video height
height: 720,
// Video frame rate
fps: 30,
// Video duration
duration: 10000,
// Output path for the video
outputPath: "./test.mp4",
// Display progress bar in the command line
showProgress: true
});

// Listen for the completion event
video.once("completed", result => {
logger.success(`Render Completed!!!\nvideo duration: ${Math.floor(result.duration / 1000)}s\ntakes: ${Math.floor(result.takes / 1000)}s\nRTF: ${result.rtf}`)
});

// Start rendering
video.start();
```

## Rendering Multi Video

```javascript
import WebVideoCreator, { VIDEO_ENCODER, TRANSITION, logger } from "web-video-creator";

const wvc = new WebVideoCreator();

// Configure WVC
wvc.config({
// Choose an appropriate encoder based on your hardware. Here, we are using the h264_nvenc encoder for Nvidia graphics cards.
// Encoder selection can refer to docs/video-encoder.md
mp4Encoder: VIDEO_ENCODER.NVIDIA.H264
});

// Create a multi-scene video
const video = wvc.createMultiVideo({
// Video width
width: 1280,
// Video height
height: 720,
// Video frame rate
fps: 30,
// Video segment parameters
chunks: [
{
url: "http://localhost:8080/scene-1.html",
// or can directly set the page content:
// content: "

Hello WebVideoCreator

",
duration: 10000,
// Insert a circular crop transition between the first and second scenes
transition: TRANSITION.CIRCLE_CROP
},
{
url: "http://localhost:8080/scene-2.html",
duration: 10000
}
],
// Output path for the video
outputPath: "./test.mp4",
// Display progress bar in the command line
showProgress: true
});

// Listen for the completion event
video.once("completed", result => {
logger.success(`Render Completed!!!\nvideo duration: ${Math.floor(result.duration / 1000)}s\ntakes: ${Math.floor(result.takes / 1000)}s\nRTF: ${result.rtf}`)
});

// Start rendering
video.start();
```

## Rendering Chunk Video and Combining into Multi Video

```javascript
import WebVideoCreator, { VIDEO_ENCODER, TRANSITION, logger } from "web-video-creator";

const wvc = new WebVideoCreator();

// Configure WVC
wvc.config({
// Choose an appropriate encoder based on your hardware. Here, we are using the h264_nvenc encoder for Nvidia graphics cards.
// Encoder selection can refer to docs/video-encoder.md
mp4Encoder: VIDEO_ENCODER.NVIDIA.H264
});

// Create chunk video 1
const chunk1 = wvc.createChunkVideo({
url: "http://localhost:8080/scene-1.html",
// or can directly set the page content:
// content: "

Hello WebVideoCreator

",
width: 1280,
height: 720,
fps: 30,
duration: 10000,
showProgress: true
});

// Create chunk video 2
const chunk2 = wvc.createChunkVideo({
url: "http://localhost:8080/scene-2.html",
width: 1280,
height: 720,
fps: 30,
duration: 10000,
showProgress: true
});

// Wait for the chunks to finish rendering
await Promise.all([chunk1.startAndWait(), chunk2.startAndWait()]);

// Set the transition effect between chunk1 and chunk2 to fade in and out
chunk1.setTransition({ id: TRANSITION.FADE, duration: 500 });

// Create a multi-scene video
const video = wvc.createMultiVideo({
width: 1280,
height: 720,
fps: 30,
// Video segments
chunks: [
chunk1,
chunk2
],
// Output path for the video
outputPath: "./test.mp4",
// Display progress bar in the command line
showProgress: true
});

// Listen for the completion event
video.once("completed", result => {
logger.success(`Render Completed!!!\nvideo duration: ${Math.floor(result.duration / 1000)}s\ntakes: ${Math.floor(result.takes / 1000)}s\nRTF: ${result.rtf}`)
});

// Start rendering
video.start();
```

## Global Configuration

You can globally configure WVC to adjust some common parameters.

```javascript
import WebVideoCreator, { VIDEO_ENCODER, AUDIO_ENCODER } from "web-video-creator";

const wvc = new WebVideoCreator();

wvc.config({
// Enable this to output some debug logs from WVC
debug: true,
// Specify the Chrome browser version to use
browserVersion: "...",
// It is recommended to enable this. If disabled, the window will be displayed and requires compatibility rendering mode to render properly. It's only used for debugging visuals.
browserHeadless: true,
// Frame rate limit is enabled by default. Disabling it can improve rendering efficiency and support animations above 60fps. However, this might disable GPU vertical sync, potentially causing screen tearing or other issues.
browserFrameRateLimit: true,
// Enable this to output browser runtime logs
browserDebug: true,
// Enable this to output each executed FFmpeg command
ffmpegDebug: true,
// Path to the FFmpeg executable file; setting this will disable the internal ffmpeg-static; it's recommended to use the internal FFmpeg for completeness
ffmpegExecutablePath: "...",
// Path to the ffprobe executable file; setting this will disable the internal ffprobe-static; it's recommended to use the internal ffprobe for completeness
ffprobeExecutablePath: "...",
// Browser GPU acceleration switch; recommended to enable for improved rendering performance; if you don't have a GPU or encounter rendering issues, you can disable it
browserUseGPU: true,
// Path to the browser executable file; setting this will disable the internal browser; it's recommended to use the internal browser for completeness
browserExecutablePath: "...",
// Allow unsafe contexts, false by default. Once enabled, users can navigate to unsafe URLs, but due to unsafe context restrictions, dynamic images and embedded videos cannot be used on the page
allowUnsafeContext: false,
// Compatible Rendering Mode is necessary on MacOS but not recommended for other environments; enabling this will disable HeadlessExperimental.beginFrame API calls and use regular Page.screenshot, which can lead to decreased rendering performance and frame rate desynchronization in some animations; you can try enabling it if you encounter the error "TargetCloseError: Protocol error (HeadlessExperimental.beginFrame): Target closed"
compatibleRenderingMode: false,
// Minimum number of browser instances in the resource pool
numBrowserMin: 1,
// Maximum number of browser instances in the resource pool
numBrowserMax: 5,
// Minimum number of page instances per browser instance
numPageMin: 1,
// Maximum number of page instances per browser instance
numPageMax: 5,
// User-agent string to use when accessing pages
userAgent: null,
// Frame capture quality (0-100); only effective for jpeg
frameQuality: 80,
// Frame image format (jpeg/png); it's recommended to use jpeg as png capture is more time-consuming
frameFormat: "jpeg",
// Timeout for capturing images with BeginFrame
beginFrameTimeout: 5000,
// Video encoder for MP4 format; by default, it uses the libx264 software encoder; it's recommended to choose the appropriate hardware encoder for acceleration based on your hardware, Encoder selection can refer to docs/video-encoder.md.
mp4Encoder: VIDEO_ENCODER.CPU.H264,
// Video encoder for WEBM format; by default, it uses the libvpx software encoder; it's recommended to choose the appropriate hardware encoder for acceleration based on your hardware
webmEncoder: VIDEO_ENCODER.CPU.VP8,
// Audio encoder; it's recommended to use the default AAC encoder
audioEncoder: AUDIO_ENCODER.AAC
});
```

## Inserting Audio

To add audio to your rendered HTML, simply include an `` element with the desired audio file. You can also set attributes like `loop`, and WVC will automatically include the audio track for looping in the video.

```html

```

You can also set various attributes to control the audio's behavior. These attributes do not always need to be paired, so you can customize them according to your needs.

```html

```

You can also dynamically add and remove `` elements in your code to control audio entering and exiting the scene. WVC will detect them.

```javascript
const audio = document.createElement("audio");
audio.src = "bgm.mp3";
// Audio enters the scene at 3 seconds
setTimeout(() => document.body.appendChild(audio), 3000);
// Audio exits the scene at 8 seconds
setTimeout(() => audio.remove(), 8000);
```

Or call [captureCtx.addAudio](./docs/capture-ctx.md#capturecontextaddaudiooptions-object) on the page to add audio to the video.

```javascript
// Add a single audio track
captureCtx.addAudio({
url: "bgm.mp3",
startTime: 500,
loop: true,
// 80% volume
volume: 80
});
// Add multiple audio tracks
captureCtx.addAudios([...]);
```

You can also directly use [addAudio](./docs/api-reference-high-level.md#singlevideoaddaudiooptions-object) in WVC to add local or remote audio to the video.

```javascript
const video = wvc.createSingleVideo({ ... });
// Add a single audio track
video.addAudio({
// url: "http://.../bgm.mp3"
path: "bgm.mp3",
startTime: 500,
loop: true,
// 80% volume
volume: 80
});
// Add multiple audio tracks
video.addAudios([...]);
```

This operation also applies to MultiVideo and ChunkVideo.

## Inserting Video

Currently, WVC supports `mp4` and `webm` video formats. To insert a video into your rendered HTML, include a `` element with the desired video file. You can set attributes like `loop` and `muted`. If your src does not contain the `. mp4` suffix name, it may not be recognized. Please add the `capture` attribute to identify the element that needs to be captured.

```html

```

You can dynamically add and remove `` elements in your code to control video entering and exiting the scene. WVC will detect them.

```javascript
const video = document.createElement("video");
video.src = "test.mp4";
// Video enters the scene at 3 seconds
setTimeout(() => document.body.appendChild(video), 3000);
// Video exits the scene at 8 seconds
setTimeout(() => video.remove(), 8000);
```

If you are using some front-end frameworks to implement animation content, WVC may not be able to listen to your changes to the `` element (such as hiding or displaying). Please replace the element with the `` element and prompt WVC to notice that it is a video canvas through `video-capture` attribute.

```html

```

### Transparent Channel Videos

Transparent videos are great for compositing digital avatars (e.g., VTubers) into video scenes. Please refer to **[Rendering Example Page](https://github.com/Vinlic/WebVideoCreator/wiki/Rendering-Example)** Last Demo

In WVC, transparent videos should be in the `webm` format. Internally, they will be re-encoded into two mp4 container videos: one for the color base video and one for the mask video. These videos will be used for blending and drawing using the `globalCompositeOperation` in the browser canvas.

For users, it's seamless. You just need to include a `` element in your HTML with the `src` set to the webm video file.

```html

```

Webm encoding and decoding can be time-consuming. If you can obtain the original mp4 video and the mask mp4 video, it's a better solution. Just add the `maskSrc` attribute.

```html

```

## Inserting Animated Images

Animated images refer to sequence frame animations in `gif` / `apng` / `webp` formats. They can naturally play in the browser, but their frame rate is usually uncontrollable. WVC proxies their rendering, replacing `img` elements with `canvas` elements, and uses ImageDecoder to decode and draw each frame in sync with virtual time.

The following animated images can be rendered as well, and you can style them as usual.

```html



```

If you are using some front-end frameworks to implement animation content, WVC may not be able to listen to your changes to the `` element (such as hiding or displaying). Please replace the element with the `` element and prompt WVC to notice that it is a dyanmic image canvas through `dyimage-capture` attribute.

```html

```

## Inserting Lottie Animations

WVC comes with the built-in [lottie-web](http://airbnb.io/lottie/#/web) animation library. If you have your own Lottie animations in your web page, they should work seamlessly with WVC.

Simply insert a `` element and set the `src` attribute.

```html

```

If you are using some front-end frameworks to implement animation content, WVC may not be able to listen to your changes to the `` element (such as hiding or displaying). Please replace the element with the `` element and prompt WVC to notice that it is a lottie canvas through `lottie-capture` attribute.

```html

```

## Applying Fonts

WVC can detect `@font-face` declarations in stylesheets and wait for the fonts to load before starting rendering.

```html

@font-face {
font-family: "FontTest";
src: url("font.ttf") format("truetype");
}

Hello World


```

Alternatively, you can register local or remote fonts through code.

```javascript
const video = wvc.createSingleVideo({ ... });
// Register a single font
video.registerFont({
// url: "http://.../font.ttf"
path: "font.ttf",
family: "FontTest",
format: "truetype"
});
// Register multiple fonts
video.registerFonts([...]);
```

Make sure the fonts can be loaded; otherwise, rendering may not start.

## Inserting Transition Effects

WVC supports the use of [Xfade](https://trac.ffmpeg.org/wiki/Xfade) filters supported by FFmpeg to create transition effects. You can refer to the [list of transitions](./docs/transition.md).

Each chunk video parameter can be configured with a transition effect and its duration.

```javascript
import WebVideoCreator, { TRANSITION } from "web-video-creator";

...

const video = wvc.createMultiVideo({
...
// Video segment parameters
chunks: [
{
url: "http://localhost:8080/scene-1.html",
duration: 10000,
// Insert a fade in/out transition between the first and second scenes
transition: {
id: TRANSITION.FADE,
duration: 500
},
// If you don't need to set the duration, you can directly set the transition ID
// transition: TRANSITION.FADE
},
{
url: "http://localhost:8080/scene-2.html",
duration: 10000
}
],
...
});

...
```

It's important to note that applying transitions will result in a shorter total video duration since the transition effect effectively overlaps a portion of two video segments. For example, if you insert a fade transition between two 5-second segments, the resulting video will have a duration of 9 seconds.

Lottie animations are also suitable for use as transition effects. You can play a full-screen Lottie animation for half of the duration at the end of one video segment and then play another full-screen Lottie animation for the remaining half at the beginning of the next video segment to create more dynamic transition effects.

## Export videos with transparent channels

WVC supports you to set the background opacity `backgroundOpacity` option to achieve transparent or semi transparent background video output, with a value range of **0-1**, Please ensure that the output video path suffix name or format option is **webm**.

```javascript
const video = wvc.createSingleVideo({
...,
// Set a completely transparent background
backgroundOpacity: 0
});
```

## Delayed Rendering Start

By default, WVC starts rendering immediately after the page navigation is complete. If you

want to perform some tasks before rendering, you can disable automatic rendering start in the options. In this case, remember to call [captureCtx.start()](./docs/capture-ctx.md#capturecontextstart) in your page code, or rendering will be blocked indefinitely.

```javascript
const video = wvc.createSingleVideo({
url: "http://localhost:8080/test.html",
width: 1280,
height: 720,
duration: 10000,
// Disable automatic rendering start
autostartRender: false
});
```

In your page code, call the start function when appropriate.

```html

// ๆ•ฐๆฎๅŠ ่ฝฝๅฎŒๆˆๅŽๅฏๅŠจๆธฒๆŸ“
loadData()
.then(() => captureCtx.start())
.catch(err => console.error(err));

```


## Start Capturing at a Specified Time Point

By default, WVC begins capturing the screen from the 0 time point after rendering starts. However, it also supports initiating screen capture from other time points.

```javascript
const video = wvc.createSingleVideo({
url: "http://localhost:8080/test.html",
width: 1280,
height: 720,
// Start capturing from the 5-second mark
startTime: 5000,
duration: 10000
});
```


## Pre-render Page Operations

```javascript
const video = wvc.createSingleVideo({
url: "http://localhost:8080/test.html",
width: 1280,
height: 720,
duration: 10000,
pagePrepareFn: async page => {
// Get the puppeteer Page object
const _page = page.target;
// Click the button
await _page.tap("#play-button");
}
});
```

## Using Action Sequences

WVC allows you to set actions to be executed at specific time points within a video. This feature enables convenient manipulation of a page at arbitrary intervals within a video. The following code is used to perform scrolling at the 3rd, 6th, and 9th seconds of the video.

```javascript
const actionFn = async (page) => {
const _page = page.target;
await _page.evaluate(() => {
window.scrollTo({
top: window.scrollY + 1280,
behavior: "smooth"
});
});
};
const video = wvc.createSingleVideo({
width: 1080,
height: 1280,
fps: 60,
outputPath: "./t2.mp4",
showProgress: true,
url: "https://www.bilibili.com/v/popular/all/",
// Setting up the action sequence
timeActions: {
3000: actionFn,
6000: actionFn,
9000: actionFn
},
duration: 10000
});
```

## Page Console Output

If you want to see the page's logs, you can enable `consoleLog` in the video options. If there are embedded videos, you can also enable `videoPreprocessLog` to output video preprocessing logs.

```javascript
const video = wvc.createSingleVideo({
...,
// Output logs from the page's console
consoleLog: true,
// Output video preprocessing logs for embedded videos
videoPreprocessLog: true
});
```

## Capture a Cover Image

After rendering a video, you can capture a frame image and save it, which can be used as the video's cover image.

```javascript
const video = wvc.createSingleVideo({
...,
// Enable cover image capture
coverCapture: true,
// Time in milliseconds to capture the image (default is 20% of the video's duration)
coverCaptureTime: 1000,
// Image format for capture (jpg/png/bmp), default is jpg
coverCaptureFormat: "jpg"
});
```

## Insert a Cover Image

WVC supports inserting an image into the first frame of the video, which will be displayed when the video is not playing.

```javascript
const video = wvc.createSingleVideo({
...,
// Set the path to the additional cover image, supports jpg/png/bmp
attachCoverPath: "./cover.jpg"
});
```

## Adjust Video Volume

You can control the overall volume of the output video.

```javascript
const video = wvc.createSingleVideo({
...,
// Set the video volume to 80% of the original
volume: 80
});
```

## Control Output Video Quality

WVC allows you to control video image quality using `videoQuality` or `videoBitrate`.

`videoQuality` is used to calculate the bitrate based on the total pixel count of the image. The following is the method used by WVC to calculate the video bitrate internally:

```javascript
const pixels = width * height;
const videoBitrate = (2560 / 921600 * pixels) * (videoQuality / 100);
```

You can provide `videoQuality` (0-100) in the video options:

```javascript
const video = wvc.createSingleVideo({
...,
// Set video quality to 80%
videoQuality: 80
});
```

If you find the bitrate inappropriate, you can set `videoBitrate` separately:

```javascript
const video = wvc.createSingleVideo({
...,
// Set video bitrate to 8Mbps
videoBitrate: "8192k"
});
```

Additionally, you can adjust the frame image quality when using jpeg as the frame format by setting `frameQuality`. For details, see [Global Configuration](#global-configuration).

You can also adjust audio quality by setting the audio bitrate `audioBitrate`:

```javascript
const video = wvc.createSingleVideo({
...,
// Set audio bitrate to 320Kbps
audioBitrate: "320k"
});
```

## Change Pixel Format

WVC currently supports output in `yuv420p`, `yuv444p`, and `rgb24` pixel formats. It defaults to using the more compatible `yuv420p`. If you notice significant color differences between the output video and the original page, you can switch to `rgb24` to improve the issue.

```javascript
const video = wvc.createSingleVideo({
...,
// Set the pixel format to rgb24
pixelFormat: "rgb24"
});
```


# Video Encoder Selection

When rendering frames from the browser and feeding them into FFmpeg, it's necessary to encode the image data at the specified frame rate and store it in a specified format container using a video encoder. Video encoding is a resource-intensive operation, and choosing a hardware encoder can accelerate this process and reduce CPU load.

Please refer to the [Video Encoder Guide](./docs/video-encoder.md) for a list of supported video encoders by WVC.


# Progress Monitoring

You can use the `progress` event of the video instance to monitor the rendering and synthesis progress.

```javascript
const video = wvc.createSingleVideo({ ... });
video.on("progress", (progress, synthesizedFrameCount, totalFrameCount) => {
// Output progress / synthesized frame count / total frame count
console.log(progress, synthesizedFrameCount, totalFrameCount);
});
```

This also applies to `MultiVideo` / `ChunkVideo` and the synthesisers in the low-level API.


# Exception Handling

## Throwing Errors

You can actively throw errors in the page to interrupt rendering.

```html

captureCtx.throwError(-1, "Abort");

```

## Listening for Page Crashes

If your page involves heavy computations or consumes excessive memory, it may crash, causing rendering to be interrupted.

If you're using the high-level API, you can be notified of page crashes through the `pageCrashed` event of the video instance.

```javascript
const video = wvc.createSingleVideo({ ... });
// Output crash errors when an error occurs
video.on("pageCrashed", err => console.error(err));
```

When using the low-level API, you can listen for page crashes through the `crashed` event of the Page instance.

```javascript
// Output crash errors when an error occurs
page.on("crashed", err => console.error(err));
```

## Listening for Other Errors

If you're using the high-level API, you can listen for errors through the `error` event of the video instance.

```javascript
const video = wvc.createSingleVideo({ ... });
video.on("error", err => console.error(err));
```

When using the low-level API, you can listen for errors through the `error` event of the Page instance.

```javascript
page.on("error", err => console.error(err));
```

## Compatible Rendering Mode

On MacOS, due to Chrome's lack of support for the BeginFrame API, it is necessary to switch to the compatible rendering mode for proper functionality. This mode may result in a decrease in rendering efficiency of around 40%. It is recommended to deploy on Windows or Linux devices for better performance.

```javascript
// Enable compatible rendering mode
wvc.config({ compatibleRenderingMode: true });
```


# Cache Management

To optimize task launching and rendering time, WVC generally maintains four types of caches: `Browser Cache`, `Preprocess Cache`, `Synthesize Cache`, and `Local Font Cache`.

**Browser Cache**: Contains cached files and records generated by the page, reducing page loading time.

**Preprocess Cache**: When rendering content references remote resources, the preprocessor attempts to fetch and cache them locally, reducing bandwidth consumption.

**Synthesize Cache**: When combining multiple video chunks into a whole, pre-rendered chunks serve as the synthesize cache. Typically, WVC clears this cache after synthesis is complete.

**Local Font Cache**: When using `registerFonts` to register local fonts for successful injection into the page, a copy of fonts from the source path is stored in the cache.

## Clearing Cache

```javascript
// Clear browser cache
wvc.cleanBrowserCache();
// Clear preprocess cache
wvc.cleanPreprocessCache();
// Clear synthesize cache
wvc.cleanSynthesizeCache();
// Clear local font cache
wvc.cleanLocalFontCache();
```


# API Reference

## High-Level API

In most cases, it is recommended to use the high-level API because it is simple but may be less flexible.

[API Reference High Level](./docs/api-reference-high-level.md)

## Low-Level API

[API Reference Low Level](./docs/api-reference-low-level.md)


# Distributed Rendering

If you have multiple devices available for rendering, you can deploy WVC on these devices. WVC provides `MultiVideo` and `ChunkVideo`, allowing you to divide the animation pages into many segments (e.g., 0-10 seconds, 10-20 seconds, etc.). Distribute their parameters to different WVC instances on different devices, create ChunkVideo instances on these devices, and execute parallel rendering to generate multiple video segments (`ts`). These segments are then sent back to the core node, where they are combined, and transitions, audio tracks, and output are handled. **The distribution and return process is not yet implemented in WVC, but it is not difficult, and you can wrap it according to your own scenario. Contributions to WVC are welcome through [PR](https://github.com/Vinlic/WebVideoCreator/pulls)!**


# Performance Tips

Performance is typically influenced by the complexity of animations and media. To improve performance:

- Divide long animations into multiple segments. For instance, you can include a seek parameter with each page URL, load the page, and seek to a specified time to start playing. Then, render and combine them as a multi-video to significantly reduce rendering time for long videos.

- Render more video chunks in parallel. To maximize system resources, select the number of parallel chunks based on the number of CPU threads, assuming your system has sufficient memory.

- CPU clock speed has a significant impact on baseline performance. Consumer-grade CPUs often have high clock speeds, which can lead to better performance.

- Consider using GPU acceleration for rendering and compositing. If your device has a GPU but it's not being utilized, check the configuration settings or report the issue.

- Using an SSD (Solid State Drive) can improve hard disk cache write performance during parallel rendering, reducing rendering time.

- Select the right video hardware encoder. By default, software encoders are used (libx264 for mp4 and libvpx for webm). If you have integrated or discrete graphics, configure the hardware encoders they support.

- Some time may be spent on network file transfers. It's advisable to deploy static file services on the same server or access the file server from a local network.

- Reducing the output video resolution and frame rate is the most effective way to reduce time consumption.

Here are the performance parameters for my personal computer as a reference:

Operating System: Windows 10 (Better performance on Linux)

CPU: AMD Ryzen 7 3700X (Clock speed 3.6-4.4GHz, 8 cores, 16 threads)

GPU: Nvidia GeForce GTX 1660 SUPER (6GB VRAM, NVENC support)

RAM: 16GB (DDR4 2400MHz)

Video Type: SVG animations + GIFs + Lottie animations

Video Resolution: 1280x720

Video Frame Rate: 30

Video Duration: 300s (5 minutes)

Rendering Time: 61s (1 minute)

Real-Time Rate: 4.844

Parallel Rendering Count: 16


# Limitations

- Constrained by browser [secure context restrictions](https://w3c.github.io/webappsec-secure-contexts/), WebVideoCreator can only access `localhost` / `127.0.0.1` or domains using HTTPS with valid certificates. For security reasons, it's recommended to use a local static server (e.g., `live-server` is a good choice), You can also use the content option to directly set the page content to avoid URL restrictions.

- The headless experimental API on Mac systems may cause crashes and needs to be switched to compatibility rendering mode to run. However, compatibility rendering mode has various issues, so it is not recommended for Mac systems. See [Compatibility Rendering Mode](#compatible-rendering-mode).

- WebVideoCreator is a pure ESM package and cannot be imported using CommonJS-style `require`. If you still want to use `require` to import it, refer to this [gist](https://gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c) for guidance.

- The page jump request during the video rendering process will be intercepted because jumping to the page will result in the loss of capture context.