Track Transformation
When transforming media, there are multiple thing that can be done for each audio or video track:
- Copying the track without re-encoding
- Re-encoding the track into a different codec
- Removing the track
@remotion/webcodecs
allows you to decide for each track what to do with it.
Using the defaults
The minimum amount of configuration is to only specify a src
and an output container
.
Using the default codecstsx
import {convertMedia } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',});
The default codecs are defined by getDefaultAudioCodec()
and getDefaultVideoCodec()
.
Choosing codecs
You can use the videoCodec
and audioCodec
options to transform all tracks to the codecs you specify.
Choosing video and audio codecstsx
import {convertMedia } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',videoCodec : 'vp8',audioCodec : 'opus',});
Handle each track individually
With the onVideoTrack
and onAudioTrack
callbacks, you can decide for each track what to do with it.
Using the onVideoTrack() APItsx
import {convertMedia } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',audioCodec : 'opus',onVideoTrack : ({track }) => {if (track .codecWithoutConfig === 'vp8') {return {type : 'copy'};}return {type : 'reencode',videoCodec : 'vp8'};},});
onVideoTrack
and onAudioTrack
have a higher priority than videoCodec
and audioCodec
.
The options for video codecs are:
{"type": "copy"}
- Copy the track without re-encoding{"type": "reencode", "videoCodec": ConvertMediaVideoCodec}
- Re-encode the track into the specified codec{"type": "drop"}
- Remove the track from the output{"type": "fail"}
- Fail and stop the conversion process
The options for audio codecs are:
{"type": "copy"}
- Copy the track without re-encoding{"type": "reencode", "audioCodec": ConvertMediaAudioCodec; bitrate: number}
- Re-encode the track into the specified codec. The suggested bitrate to use is128000
.{"type": "drop"}
- Remove the track from the output{"type": "fail"}
- Fail and stop the conversion process
The enums ConvertMediaVideoCodec
and ConvertMediaAudioCodec
can be imported from @remotion/webcodecs
.
Checking if a track can be copied
To check if it is possible to return {"type": "copy"}
, you can use canCopyTrack
property you get from onVideoTrack
.
Using the canCopyVideoTrack() APItsx
import {convertMedia } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',audioCodec : 'opus',onVideoTrack : ({track ,inputContainer ,outputContainer ,canCopyTrack }) => {if (canCopyTrack ) {return {type : 'copy'};}return {type : 'reencode',videoCodec : 'vp8'};},});
To check outside of a onVideoTrack
handler, you can also use the the canCopyVideoTrack()
and canCopyAudioTrack()
APIs
Checking if a track can be re-encoded
To check if it is possible to return {"type": "reencode"}
, you can use the canReencodeVideoTrack()
and canReencodeAudioTrack()
APIs.
Using the canReencodeVideoTrack() APItsx
import {convertMedia ,canReencodeVideoTrack } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',audioCodec : 'opus',onVideoTrack : async ({track }) => {constcanReencode = awaitcanReencodeVideoTrack ({videoCodec : 'vp8',track ,resizeOperation : null,rotate : 0,});if (canReencode ) {return {type : 'reencode',videoCodec : 'vp8'};}return {type : 'drop'};},});
Asynchronously determining action
The onAudioTrack
and onVideoTrack
callbacks can be asynchronous.
While the operations are unresolved, reading of the input fill is paused.
Decide behavior upfront
If you want to display a UI letting the user choose codec settings before the conversion starts, you can do so.
Use parseMedia()
to get video and audio tracks respectively:
Using parseMedia() to get tracks upfront.tsx
import {parseMedia } from '@remotion/media-parser';const {tracks } = awaitparseMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',fields : {tracks : true,},});
You now have an object of two arrays of VideoTrack
and AudioTrack
objects.
You can now use canReencodeAudioTrack()
, canReencodeVideoTrack()
, canCopyAudioTrack()
, and canCopyVideoTrack()
to determine which options to show.
Use the onVideoTrack
and onAudioTrack
callbacks to return the user selection.
You can use the trackId
field as the unique key for each track.
Falling back to default
The default values for onVideoTrack
and onAudioTrack
are the functions defaultOnVideoTrackHandler
and defaultOnAudioTrackHandler
respectively.
If you only want to override part of the logic, you can return the default resolver functions at the end of your logic.
Falling back to the default behaviortsx
import {convertMedia ,defaultOnAudioTrackHandler } from '@remotion/webcodecs';awaitconvertMedia ({src : 'https://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4',container : 'webm',onAudioTrack : (params ) => {// Custom logic for handling video tracks// ...// Fall back to the default behaviorreturndefaultOnAudioTrackHandler (params );},});
Debugging
Pass logLevel: "verbose"
to convertMedia()
to see debug information in the console, including how the defaults have decided which operations to take.
Reference implementation
Visit the source code for convert.remotion.dev to see a reference implementation for an online video converter that displays a user interface for all possible options.