TinyVGM.js is a JavaScript library for parsing VGM (Video Game Music) files. It provides tools to read and interpret the header, metadata, and commands from VGM files, making it easier to work with this audio file format in JavaScript environments.
See the C implementation of TinyVGM
npm install tinyvgm
Here's a basic example of how to use TinyVGM.js to parse a VGM file:
import { parseVGM } from 'tinyvgm'
// Assuming you have a VGM file loaded into an ArrayBuffer 'vgmBuffer'
const context = parseVGM(vgmBuffer)
// Accessing header, metadata, and commands
for (const field of context.header) {
console.log(`Header Field: ${field.type}, Value: ${field.data}`)
}
// Displaying extra header information, if available
if (context.extraHeader) {
console.log('Extra Header:', context.extraHeader)
}
if (context.metadata) {
for (const metaField of context.metadata) {
const text = new TextDecoder('utf-16le').decode(metaField.data)
console.log(`Metadata Type: ${metaField.type}, Value: ${text}`)
}
}
for (const command of context.commands) {
console.log(`Command: ${command.cmd}`)
if (command.data) {
console.log(`Data:`, command.data)
}
}
The context object returned by the parseVGM
function, containing parsed data from the VGM file.
- version: VGM file version.
- loopCount: Number of times to loop playback. Change this value to
0
to disable loop while playing. - hasLoop: Indicates if the file has a loop point.
- totalSamples: Total number of samples in the file. One sample is 1/44100 seconds.
- loopSamples: Number of samples in the loop. One sample is 1/44100 seconds.
- skipUnknownCommand: Skip unknown commands during parsing.
- header: Function that returns a generator for header information.
- extraHeader: Function that returns extra header information, if available.
- metadata: Function that returns a generator for metadata, if available.
- commands: Function that returns a generator for command data, accepting an optional loops parameter to set the number of loops.
- samplesPlayed: Number of samples played so far.
- loopSamplesPlayed: Number of loop samples played in the current loop.
- onLoop: Function called when a loop point is reached, with the remaining loop count as a parameter.
Options for parsing the VGM file.
- loopCount: Optional. How many times to loop the playback. Set
Infinity
to loop indefinitely. - skipUnknownCommand: Optional. Whether to skip unknown commands during parsing.
- onLoop: Optional. Function to be called when a loop point is reached, with the remaining loop count as a parameter.
- If the VGM file does not contain loop by default, the whole song will be looped if
loopCount
is set.
Parses a VGM file and returns a context object containing the parsed data.
- buffer: The VGM file data as an
ArrayBuffer
orDataView
. - options: Optional. Configuration options for parsing.
A utility class for creating enumerative types.
add(key: string, id?: number)
: Adds a new enumeration key.
VGM_CMD_LENGTH_TABLE
: Array representing the lengths of various VGM commands.TinyVGMHeaderField
: Enumeration of VGM header fields.TinyVGMMetadataType
: Enumeration of VGM metadata types.
The header
generator yields information about the VGM file header. Each iteration returns an object with type
and data
properties.
import { parseVGM } from 'tinyvgm'
// Load your VGM file into an ArrayBuffer `vgmBuffer`
const { header } = parseVGM(vgmBuffer)
for (const field of header()) {
console.log(`Header Field: ${field.type}, Value: ${field.data}`)
}
The metadata
generator yields metadata from the GD3 tag of the VGM file, if present. Each iteration provides an object with type
and data
properties, where data
is a Uint8Array
representing a string in UTF-16.
import { parseVGM } from 'tinyvgm'
const { metadata } = parseVGM(vgmBuffer)
if (metadata) {
const decoder = new TextDecoder('utf-16le')
for (const metaField of metadata()) {
// Convert Uint8Array to a UTF-16 encoded string
const text = decoder.decode(metaField.data)
console.log(`Metadata Type: ${metaField.type}, Value: ${text}`)
}
}
The commands
generator yields the commands found within the VGM file. Each command is represented as an object containing cmd
and optionally type
and data
properties.
import { parseVGM } from 'tinyvgm'
const { commands } = parseVGM(vgmBuffer)
for (const command of commands()) {
if (command.cmd === 0x67) {
// Handling a data block
console.log(`Data Block Detected`)
// The type of the data block is indicated by the byte following the 0x66 marker
if (command.type !== undefined) {
console.log(`Data Block Type: ${command.type}`)
}
// The actual data of the data block
if (command.data) {
console.log(`Data Block Length: ${command.data.length}`)
// Process the data as needed
}
} else {
// Handling other commands
console.log(`Command: ${command.cmd}`)
if (command.data) {
console.log(`Data:`, command.data)
}
}
}
- VGZ files should be decompressed before passing to
parseVGM
. - The usage of these generators assumes that you have already loaded a VGM file into an
ArrayBuffer
namedvgmBuffer
. - The generators yield data lazily, meaning they only process parts of the VGM file as you iterate over them. This can be efficient for large files.
- The
metadata
generator is optional and will only be present if the GD3 tag is found in the VGM file. - The interpretation of the yielded data (especially command data) will depend on your specific requirements and the VGM file format specification.
Contributions are welcome! Please feel free to submit pull requests or create issues for bugs, feature requests, or improvements.
TinyVGM.js is released under the MIT License.