A library to create WebVTT (and SRT) files
This repo contains a library to create subtitle-files in the WebVTT and SRT formats which we created for audapolis.
npm install @audapolis/webvtt-writer
First, you have to create a WebVtt
-Instance.
This represents one subtitle file.
import { WebVtt } from '@audapolis/webvtt-writer';
const minimal_vtt = new WebVtt();
You can optionally give it a text header:
const vtt = new WebVtt('Subtitles are cool');
Now you can start adding elements to you subtitle file:
Comments are an optional component that can be used to add information to a WebVTT file. Comments are intended for those reading the file and are not seen by users.1
To add a comment you have to create a new VttComment instance and then add it to the WebVtt object:
import { VttComment } from '@audapolis/webvtt-writer';
const myComment = new VttComment('This is a comment\nIt can even be multiline');
vtt.add(myComment);
Cues are subtitle lines. Each cue has a start and end time and a payload -- the text that should be displayed. It may optionally also have
- an identifier
- cue settings
Since every cue needs a start time, an end time and a payload, this is one of the simplest cues possible:
import { VttCue } from '@audapolis/webvtt-writer';
const mySimpleCue = new VttCue({
startTime: 1,
endTime: 2,
payload: 'This will be shown\nincluding this second line',
});
vtt.add(mySimpleCue);
Start and end time are given in seconds.
A cue may also have an identifier which can be used to reference this cue, for example for styling it. They do not have to be unique, but are often numbered.
const cueWithIdentifier = new VttCue({
startTime: 3,
endTime: 4,
payload: 'This cue has an identifier',
identifier: "Hey, i'm an identifier",
});
vtt.add(cueWithIdentifier);
Cue settings are used to position the cue payload text. There are 5 cue settings, whose meanings and possible values are described here:
- vertical
- line
- position
- size
- align
All of them are optional but if you want to, you can specify them like this:
import { VttCueSettings } from '@audapolis/webvtt-writer';
const mySettings = new VttCueSettings({ size: '50%' });
const cueWithSettings = new VttCue({
startTime: 5,
endTime: 120,
payload: 'This cue has settings',
settings: mySettings,
});
vtt.add(cueWithSettings);
The payload may not contain the characters -
and <
and the usage of >
is not recommeneded. Instead their escape
sequences should be used. Because of that webvtt-writer
automatically escapes the payload and identifier.
However this means that you cannot use a payload which contains cue payload text tags,
as they would be escaped as well.
If you want to use payload tags, you can set payloadEscaped
to true
. This will disable automatic payload escaping.
Warning: It is now your task to escape the payload text, so use escapeVttString
widely.
For example:
import { escapeVttString } from '@audapolis/webvtt-writer';
const cueWithoutEscaping = new VttCue({
startTime: 130,
endTime: 200,
payload: `<b>${escapeVttString('This cue')}</b>${escapeVttString(' has cue text tags')}`,
payloadEscaped: true,
});
vtt.add(cueWithoutEscaping);
To get the content of the subtitle file as a string, you can use .toString()
on you WebVtt
object:
const vttString = vtt.toString();
console.log(vttString);
From the examples above, you should get:
WEBVTT Subtitles are cool
NOTE This is a comment
It can even be multiline
00:00:01.000 --> 00:00:02.000
This will be shown
including this second line
Hey, i'm an identifier
00:00:03.000 --> 00:00:04.000
This cue has an identifier
00:00:05.000 --> 00:02:00.000 size:50%
This cue has settings
00:02:10.000 --> 00:03:20.000
<b>This cue</b> has cue text tags
This can now be written to a file, for example on nodejs like this:
fs.writeFileSync('subtitles.vtt', vttString);
You can also export the subtitles as an SRT file, which has broader software support. Note: Srt does not support some features of WebVTT. We know of at least:
- header (including header text)
- cue text tags
- cue settings
- comments
The header, cue settings and comments will simply be ignored when exporting to srt. Since cue text tags are not handled by this library, you have to take care to not add them if you want to create an srt file.
Knowing all this, you can render the srt version of using .toString('srt')
:
const srtString = vtt.toString('srt');
console.log(srtString);
This should give you:
00:00:01.000 --> 00:00:02.000
This will be shown
including this second line
Hey, i'm an identifier
00:00:03.000 --> 00:00:04.000
This cue has an identifier
00:00:05.000 --> 00:02:00.000
This cue has settings
00:02:10.000 --> 00:03:20.000
<b>This cue</b> has cue text tags