-
Notifications
You must be signed in to change notification settings - Fork 0
06. Visualization Configuration
The visualization configuration tells monkey-hi-hat what kind of data to send to the shaders, what kind of audio textures are used (if any), and where to find the vertex and shader files. There are many examples in my Volt's Laboratory repository, and I'll be adding to those over time.
-
Description
A line of text with the name of the visualization, and anything else you'd like to add such as the author's name or a URL (remember, no hashtags, that signifies the beginning of a comment). The monkey-droid GUI control app will read and show these descriptions. -
VertexSourceTypeName
This controls how data is fed to the shaders. Currently two built-in types are supported:VertexQuad
andVertexIntegerArray
. See Creating Visualizations for more information about these. -
VertexShaderFilename
andFragmentShaderFilename
are the names of the vert and frag shaders. They must be in the same directory as the.conf
file. If one or the other is omitted, a pass-through default is provided automatically. -
BackgroundFloatRGB
is the background color the buffer is cleared to before each frame is rendered. This is more important withVertexIntegerArray
because the output doesn't typically cover the entire screen, unlikeVertexQuad
which is full-coverage by definition. As the name suggests, the red, green, and blue color components must be normalized floats (values 0 to 1). -
RenderResolutionLimit
is disabled by default (value 0), but it allows you to apply a maximum resolution to the rendered output, which will then be upscaled to full-screen. This is useful for heavy-overhead shaders (like cloud_tunnel) that can't sustain smooth frame rates at larger sizes. -
FXResolutionLimit
is disabled by default (value 0). It allows you to apply a maximum rendering resolution when a post-processing FX is active. The FX configuration must also set theApplyPrimaryResolutionLimit
flag, indicating the FX is high-overhead, otherwise this value is not used. When the program is cross-fading between two visualizations, this value is also applied if it is defined, regardless of whether any FX is running. -
RandomTimeOffset
is disabled by default (value 0). It will add or subtract a randomly-generated offset to thetime
uniform passed to shaders. This is useful when a shader is highly time-dependent, but produces very different output over time. Note that some shaders need to "seed" their drawing buffers for the first few passes. If they rely ontime
for this, you may be able to convert them to use theframe
count instead (especially if you have defined a frame-rate target).
Vertex Source Types can have their own custom settings. The section name matches the Visualizer Type Name.
Currently only VertexIntegerArray
has additional settings:
-
VertexIntegerCount
defines the size and content of the integer array passed to the vertex shader. This must be a 32-bit integer value, so the visualizer can request one thousand, ten thousand, a million, or any other number of values. Typically, very high values aren't that useful, and the VertexShaderArt website limits users to a maximum value of 100,000. -
ArrayDrawingMode
defines how the output vertices are drawn. These are defined by OpenGL and include the following settings:Points
Lines
LineStrip
LineLoop
Triangles
TriangleStrip
TriangleFan
If the visualizer is audio-responsive, this section lists the uniform names of audio textures that will be used by the shaders. Unlike the first version, you must use these pre-defined uniform names in this file and in your shaders. See Understanding Audio Textures for more details, but the available types and names are:
AudioTexture typename | Fixed uniform name |
---|---|
AudioTextureWaveHistory | eyecandyWave |
AudioTextureFrequencyDecibelHistory | eyecandyFreqDB |
AudioTextureFrequencyMagnitudeHistory | eyecandyFreqMag |
AudioTextureWebAudioHistory | eyecandyWebAudio |
AudioTextureShadertoy | eyecandyShadertoy |
AudioTexture4ChannelHistory | eyecandy4Channel |
AudioTextureVolumeHistory | eyecandyVolume |
As of version 2.0, it is no longer necessary to consider or manage OpenGL TextureUnit
assignments. Also, the [AudioTextureMultipliers]
section is obsolete. You should apply any boost/cut multipliers in the shader code, or as custom uniforms in the visualizer configuration file (see below).
Custom fixed-value or random-value float
uniforms can be defined. These are passed to all shaders used by the visualizer. The format is uniform=value
or uniform=min:max
. Randomized uniforms are generated each time the config file is loaded, and will remain fixed during that specific run.
Note that due to mathematical quirks of the IEEE double
data type, the true maximum value will always be very slightly smaller than the requested value (specifically, 0.0000000004656612875245796924106 less... likely irrelevant for visualization purposes).
See also Shader Basics for other uniforms provided by the program such as time
and frame
and several random numbers.
It's common for shaders to reference a static image as input data. These can be loaded into memory and provided to all shaders as sampler2D
uniforms. The format is uniform:filename.ext
and all texture paths defined in mhh.conf
will be searched. As of v3.1.1, the program can randomly select a texture. Specify the same uniform name multiple times with a different texture filename on each line.
The image-loading library (StbImageSharp) supports the JPG, PNG, BMP, TGA, PSD, GIF and HDR formats, although I have only tested with JPG and PNG files.
The Volt's Laboratory archive file provided with each release includes the textures available at the Shadertoy website.
The version 3.0 post-processing FX shaders sometimes don't work well (and sometimes don't work at all) with certain visualizations, so this section lets a visualizer list the names of any FX shaders that should never be applied. It's just a simple list of the FX config filenames.
A future release will allow you to specify * which blocks all FX processing. This is useful for visualizations that should only be loaded with specific FX shaders (and never without), since playlists and the command-line can specify a visualizer / FX combination directly.
Instead of drawing directly to the OpenGL swap-buffers, a multi-pass shader draws to internally-managed framebuffers. A previously-used framebuffer's texture can be used as input to later passes. Each "pass" consists of another visualizer input object and a pair of vertex and fragment shader files. Other than actually writing the shaders, your main task is to figure out the sequence of buffer usage to produce the desired effect. The buffer numbers are zero-based and should increase sequentially. Skipping numbers or trying to use buffer numbers as inputs that haven't been defined yet will prevent the visualizer from loading.
Each entry under the [multipass]
section header defines an additional rendering pass (or "draw call" in OpenGL terminology). The final draw buffer is then automatically copied to OpenGL's internal "back buffer" then swapped to the screen. Each pass is defined by four to six space-separated columns:
# | Content | Description |
---|---|---|
1 | draw buffer | A single zero-based integer |
2 | input buffers | An asterisk indicating none, or a comma-separated list of previously-used draw buffers |
3 | vertex | The name of a vertex shader source file, or asterisk to use the one in the [shader] section |
4 | fragment | The name of a fragment shader source file, or asterisk to use the one in the [shader] section |
5 | vertex source | Optional; blank, or VertexIntegerArray or VertexQuad
|
6 | settings | Required for VertexIntegerArray to specify the count and draw modes (see below) |
For more complex passes, it is possible to specify visualizer .conf
files instead of shaders and vertex types:
# | Content | Description |
---|---|---|
1 | draw buffer | A single zero-based integer |
2 | input buffers | An asterisk indicating none, or a comma-separated list of previously-used draw buffers |
3 | viz.conf | The name of a visualizer .conf file, or asterisk to use the "main" visualizer file |
Buffers are just single numbers. You can draw to the same buffer multiple times within the visualizer, although you can't simultaneously use the same one to draw and as an input on any single pass. Unless you specify a RenderResolutionLimit
, remember that each buffer is the size and color-depth of your display area, so if you're rendering something like 32bpp at 2K, these will add up quickly (not to mention overhead allocating them, shuffling them around, and so on). Also bear in mind that during the transition to a new visualizer, both visualizers and all their buffers will be allocated, as well as two additional buffers used by the crossfade shader. It isn't hard to imagine a high-res display rapidly depleting GPU memory, so reusing buffers is a smart optimization.
The texture of any draw buffer used in a previous pass can be specified in a comma-delimited list. The uniforms are always named input0
, input1
and so on, where the number at the end matches the buffer number. So if the pass specifies 1,3
the shader will receive textures in uniforms named input1
and input3
. If no inputs are required (commonly on the first pass), use an asterisk in this column.
The repository's test-content multipass visualizer is an example of this (it is the same as the oil_slick visualization in Volt's Laboratory and the content archive provided in the release downloads).
The final texture of any draw buffer from the previous frame can be specified in a comma-delimited list along with the numeric input buffers (numeric buffers refer to the buffer contents from the current frame). The program refers to previous-frame content as "back-buffers" although that's a bit different than what back/front buffer normally means. So front- or draw-buffer 0 corresponds to back-buffer A, and 1 is B, 2 is C, and so on. Their uniform names will be inputA
, inputB
and so on.
Unlike Shadertoy buffers, the contents of the back-buffers is fixed throughout the frame. This is actually increased flexibility. Shadertoy buffers only temporarily have access to the previous frame's data -- new output overwrites that data and it isn't available to later passes in the frame. This will be discussed more in the Shader Basics section.
The repository's test-content doublebuffer visualizer is an example of this (it is the same as the darkstar visualization in Volt's Laboratory and the visualizer archive provided in the release downloads).
For the simplified multipass definition, provide the names of complete visualizer .conf
files. The repository's test-content mpvizconf visualizer is an example of this. Use an asterisk in this column to use the main (initial) visualizer configuration where the multipass is defined.
If a visualizer config is defined, no additional columns are required or allowed.
This is pretty self-explanatory -- provide the names of shader files. The paths defined in the mhh.conf
configuration will be searched. The .vert
and .frag
extensions are optional and will be added if needed. To use the shader files defined in the [shader]
section, use an asterisk in either or both columns. This is common for the first pass where the basic visualizer's shader filenames are both used.
These columns are optional. If blank, the pass will re-use the vertex data object defined in the basic [shader]
section. Otherwise either of the available visualizers can be declared: VertexIntegerArray
or VertexQuad
.
The settings column is required when VertexIntegerArray
is used since it is currently the only visualizer that accepts settings. The settings entry must specify both VertexIntegerCount
and ArrayDrawingMode
separated by a semicolon. See the earlier sections above for details about those settings. The column should look like this:
VertexIntegerCount=1000;ArrayDrawingMode=Triangles