ImpVis is a real-time visualization tool for displaying isosurfaces and scalar fields of algebraic and non-algebraic 3D implicit functions. The application focuses on interactivity, allowing users to change the function parameters and expressions on the fly.
Below is a screenshot of ImpVis displaying a quintic surface known as Togliatti surface. The positive and negative sides of the surface are shown in red and magenta, respectively.
ImpVis can also render scalar fields using direct volume rendering, as shown below. The color transfer function maps positive values to red/orange hues, negative values to magenta/purple, and values near zero to white. The mapping can be adjusted using an exponential scaling factor.
- Drag to rotate the surface (left mouse button) or light source (right mouse button).
- Use the mouse wheel to zoom in/out.
- Press F11 to toggle fullscreen.
- Drag the bottom slider to set the isovalue.
- Use the top-right window to select an equation, change the function parameters and adjust the render settings.
ImpVis features an equation editor that allows users to write new expressions by modifying existing ones from a catalog of predefined implicit equations.
In the equation editor, the expression for the left-hand side of the equation is written in a modified GLSL ES syntax in which the caret symbol ^
is used as an exponentiation operator instead of bit-wise exclusive or operator. For example, for the Torus implicit equation
the left-hand side expression can be written as (c-sqrt(x^2+y^2))^2+z^2-a^2
instead of pow(c-sqrt(x*x+y*y),2.0)+z*z-a*a
, which is also supported.
Under the hood, exponentiation expressions of the form b^n
, where n
is a positive integer between 2 and 16 (inclusive), are converted to the product of multiplying n
bases: b*b*...*b
. This is often more efficient than using the native pow
function. ImpVis also has custom built-in functions mpow2(b)
, mpow3(b)
, up to mpow16(b)
, that can be used in place of b^2
, b^3
, up to b^16
, where b
is an expression that evaluates to float
.
Another syntax difference from GLSL is the use of square brackets for grouping (as parentheses) instead of array indexing. For example, for the Cassini quartic surface equation
the left-hand side expression can be written as [(x+r)^2+y^2]*[(x-r)^2+y^2]-z^2
.
The editor also allows injecting GLSL ES code both in the global scope of the shader and in the local scope of the shader function that evaluates the implicit function. Try selecting different equations from the catalog to see how their scopes are implemented in the equation editor. As a rule of thumb:
- Use the global scope to define GLSL functions and constants to be used either in the local scope or directly in the implicit function.
- Use the local scope to define GLSL variables to be used in the implicit function expression.
Any changes to the currently selected equation (expression, local scope, or global scope) will be saved as a user-defined equation at the end of the group of equations called "Other".
The following apply for code injected in the local scope:
p
is the name of avec3
variable containing the values of thex
,y
, andz
variables of the 3D implicit function. Thus, in the local scope, usep.x
,p.y
, andp.z
instead ofx
,y
, andz
.p2
is a shortcut forp*p
,p3
is a shortcut forp*p*p
, and so on up top16
. Again, these are often more efficient thanpow(p,n)
. The functionsmpow2(b)
,mpow3(b)
, up tompow16(b)
, are also available in the local scope.
The isosurfaces are rendered using an adaptive raymarching algorithm. The scalar fields are rendered using direct volume rendering. Both are implemented as GLSL ES 3.00 shaders. The equation names and the expressions shown in the top-left corner of the screen are rendered using MathJax (WebAssembly only).
The adaptive raymarching algorithm adjusts the size of the ray's next step according to the value of the scalar field and gradient evaluated at the current step. The step size decreases as the ray approaches the surface, and increases as it moves away from it. This is similar to the adaptive marching points algorithm described in Real-Time Ray Tracing of Implicit Surfaces on the GPU (Singh et al. 2009). However, in ImpVis the step size varies gradually as the rays approach the surface, thus reducing the number of conditional branchings. In addition, the step size decreases as the ray approaches the limits of the bounding volume as a measure to avoid clipping artifacts. For details, read the inline comments in the fragment shader at src/assets/shaders/raycast.frag
.
ImpVis can be built for the desktop (Windows, Linux, macOS) and the web (WebAssembly).
First clone the repo:
git clone /~https://github.com/hbatagelo/impvis.git
cd impvis
Make sure the following tools are installed and are reachable from the path:
- Conan >=1.44, <2.0 (not required for WebAssembly). Conan 2.* is not supported yet.
- CMake 3.18 or later.
- A C++ compiler with at least partial support to C++20 (tested with GCC 11, Clang 13, MSVC 17, and emcc 3.1).
Run build-vs.bat
for building with Visual Studio 2022, using x64 as target platform. The script will execute the following commands:
:: Create the build directory
mkdir build && cd build
:: Set the build type
set BUILD_TYPE=Release
:: Configure (VS 2022 generator for a 64-bit target architecture)
cmake -G "Visual Studio 17 2022" -A x64 -DCMAKE_BUILD_TYPE=%BUILD_TYPE% ..
:: Build
cmake --build . --config %BUILD_TYPE%
Run ./build.sh
to execute the following commands:
# Create the build directory
mkdir build && cd build
# Set the build type
BUILD_TYPE=Release
# Configure (Makefile generator)
cmake -DCMAKE_BUILD_TYPE=$BUILD_TYPE ..
# Build
cmake --build . --config $BUILD_TYPE
Note: on macOS, if you encounter errors building with AppleClang, try Clang installed via HomeBrew.
- Install Emscripten and activate its environment variables.
- Run
build-wasm.bat
(Windows) orbuild-wasm.sh
(Linux, macOS).
The WASM binaries will be written to impvis/public
.
- Configure CMake with
-DENABLE_UNIT_TESTING=ON
(default isOFF
). - Build the project. The executable
tests
will be written to the build directory. - Run
tests
.
-
Install libFuzzer.
-
Configure CMake with Clang as compiler, and
-DENABLE_FUZZ_TESTING=ON
(default isOFF
). -
Build the project. The executable
fuzzer
will be written to the build directory. -
Copy
fuzzer
totests/fuzzer
and run:./fuzzer corpus -dict=dictionary.txt -max_len=1280 -timeout=5
This will use the corpus of sample inputs contained in
tests/fuzzer/corpus
, and the dictionary of keywords intests/fuzzer/dictionary.txt
. During the fuzzing process, test cases that trigger coverage of new paths through the code will be added to the corpus directory.By default, the fuzzer runs indefinitely, or until a bug is found. Use the parameter
-max_total_time
to set a time limit. For example,-max_total_time=60
forces the test to stop after one minute.
The equations are described as TOML files located in src\assets\equations
.
Each TOML file must have a title
key in root level with a value that is the name of the equation group (e.g., title="Cubic"
for the file that contains cubic equations). Each equation is then described by a table with the following key/value pairs, from which only the expression
key/value pair is required:
Key | Value type | Description |
---|---|---|
name |
string | Equation name |
expression |
string | Left-hand side of the equation expression |
thumbnail |
string | Path to the thumbnail file, relative to /src/assets |
bounds_shape |
integer (0 or 1 ) |
Bounding shape (0 : sphere, 1 : box) |
bounds_radius |
positive float | Bounding radius |
raymarch_method |
integer (0 or 1 ) |
Ray marching method (0 : adaptive, 1 : fixed-step) |
raymarch_steps |
positive integer | Maximum number of ray marching steps |
raymarch_root_test |
integer (0 or 1 ) |
Ray marching root test (0 : sign change, 1 : Taylor 1st-order) |
camera_distance |
positive float | Initial distance from camera |
colormap_scale |
positive float | Colormap scaling factor for direct volume rendering |
parameters |
array of inline tables | Expression parameters (see details below) |
code_local |
string | GLSL code to be injected in the local scope |
code_global |
string | GLSL code to be injected in the global scope |
comment |
string | Comments in LaTeX math mode |
Each inline table of the parameters
' array must have two key/value pairs:
Key | Value type | Description |
---|---|---|
name |
string | Parameter name |
value |
number | Parameter value |
Below is an excerpt from 01_quartic.toml
. Browse src\assets\equations
for more examples.
title = "Quartic"
[bifolia]
name = "Bifolia"
thumbnail = "textures/thumbs/quartic/bifolia.png"
bounds_radius = 2
raymarch_steps = 80
camera_distance = 7
colormap_scale = 7
expression = "(x^2+y^2+z^2)^2-3*y*(x^2+z^2)"
[cassini]
name = "Cassini"
thumbnail = "textures/thumbs/quartic/cassini.png"
bounds_radius = 3
raymarch_steps = 75
camera_distance = 13
colormap_scale = 0.75
parameters = [{name = "r", value = 1}]
expression = "[(x+r)^2+y^2]*[(x-r)^2+y^2]-z^2"
[chair]
name = "Chair"
thumbnail = "textures/thumbs/quartic/chair.png"
bounds_radius = 2
raymarch_steps = 100
camera_distance = 7
colormap_scale = 8
parameters = [{name = "a", value = 0.95},
{name = "b", value = 0.8},
{name = "k", value = 1}]
expression = "(x^2+y^2+z^2-a*k^2)^2-b*[(z-k)^2-2*x^2]*[(z+k)^2-2*y^2]"
[chmutov_quartic]
name = "Chmutov Quartic"
thumbnail = "textures/thumbs/quartic/chmutov.png"
bounds_shape = 1
bounds_radius = 1.2
raymarch_steps = 70
camera_distance = 8
colormap_scale = 0.75
code_global = '''
float T_4(float x) {
float x2=x*x;
float x4=x2*x2;
return 8.*x4-8.*x2+1.;
}
'''
comment = '''
,\\&\mathrm{where}\; T_4(x)=8x^4-8x^2+1\\
&\mathrm{is\;the\;Chebyshev\;polynomial\;of\;the\;first\;kind\;of\;degree\;4.}
'''
expression = "T_4(x)+T_4(y)+T_4(z)"
MIT License.
ImpVis was originally designed in 2014, based on Qt, and licensed under GPLv3. The current version is a complete rewrite of the first version. It is now available under the MIT license and is based on the ABCg framework.