Lindsay Kay
BioDigital Tech Talk
July 14, 2017
Silhouettes
Emphasis
for
in
WebGL
data:image/s3,"s3://crabby-images/88a31/88a316902b85d5905794bb9234bca52e1c4144bb" alt=""
data:image/s3,"s3://crabby-images/28c75/28c75149921e71b322e7d905c639f939b44a7ce3" alt=""
Silhouettes: Stencil Technique
- Draw non-emphasized objects
- Enable stencil testing
- Draw emphasized objects
- Configure to not render on top of pixels already in stencil buffer
- Draw emphasized objects again, with flat color, expanding mesh along vertex normals
data:image/s3,"s3://crabby-images/1f805/1f805d540a25992997b0b152bc4ade323e52cb19" alt=""
data:image/s3,"s3://crabby-images/cf1c5/cf1c53bc86dc36b05618e5a460067777551294bc" alt=""
Step (3)
Step (5)
Combined result
Silhouettes: Stencil Technique (2)
We expand along the normals because simply scaling an object will create a uneven silhouette:
data:image/s3,"s3://crabby-images/58731/58731f2bf5bded20f29032ea919b6cb909475b09" alt=""
Silhouettes: a small enhancement
data:image/s3,"s3://crabby-images/2b11f/2b11fe68daf81eca87740e79908d2fcb47ea731b" alt=""
data:image/s3,"s3://crabby-images/5991a/5991ad19a3a48cdd44bba39a00b451ca19a7dacb" alt=""
data:image/s3,"s3://crabby-images/9c8c6/9c8c6befb675299231e53a640cdd290f4cc87ad8" alt=""
Expand each outline vertex by amount proportional to its depth
Silhouettes: The JavaScript
// Render object using standard shader
gl.clearStencil(0);
gl.clear(GL_STENCIL_BUFFER_BIT);
gl.enable(GL_STENCIL_TEST);
gl.stencilFunc(GL_ALWAYS, 1, -1);
gl.stencilOp(GL_KEEP, GL_KEEP, GL_REPLACE);
drawObject();
// Render glowing, expanded version of object
gl.stencilFunc(GL_NOTEQUAL, 1, -1);
gl.stencilOp(GL_KEEP, GL_KEEP, GL_REPLACE);
drawObject();
Silhouettes: The GLSL
attribute vec4 position;
uniform mat4 modelMat;
uniform mat4 viewMat;
uniform mat4 projMat;
attribute vec3 normal;
uniform float outlineWidth;
void main(void) {
vec4 projPos = projMat * viewMat * modelMat * vec4(position.xyz, 1.0);
vec3 offset = (normal * (outlineWidth * (projPos.z/1.0)));
vec4 worldVertex = modelMat * vec4(modelVertex.xyz + offset, 1.0);
mat4 viewMatrix = viewMat;
gl_Position = projMat * (viewMatrix * worldVertex);
}
uniform vec3 outlineColor;
void main(void) {
gl_FragColor = vec4(outlineColor, 1.0);
}
Vertex shader:
Fragment shader:
Silhouettes: a small hitch for some objects
data:image/s3,"s3://crabby-images/e6ce4/e6ce4a418f9d48e84ffffe4460595952e3946066" alt=""
- Mesh normals must be smooth, otherwise there will be gaps in the silhouette at hard edges
- Not a problem for organic shapes, though
data:image/s3,"s3://crabby-images/28c75/28c75149921e71b322e7d905c639f939b44a7ce3" alt=""
Silhouettes: Drawing halo as wireframe
- Draw halo as thick wireframe
- Solves broken outline on hard edges
- Problem: WebGL on Windows has limited wire thickness
data:image/s3,"s3://crabby-images/6dfc1/6dfc1994e87188c233ebf8a9376523eafcba33c5" alt=""
data:image/s3,"s3://crabby-images/cf1c5/cf1c53bc86dc36b05618e5a460067777551294bc" alt=""
Standard shader (1)
Thick wireframe (5)
Result
Silhouettes: Status in Human
data:image/s3,"s3://crabby-images/5d136/5d1369d6e588cbdb0ed95747e01f6b1e9e9c0d7f" alt=""
- Implemented in BDS fork of SceneJS
- https://github.com/biodigital-human/scenejs/tree/outlining
- Using vertex displacement method, breaks on hard edges
- Need to find workaround for bug in WebGL before deploying
Outlining for Emphasis in WebGL
By xeolabs
Outlining for Emphasis in WebGL
A simple WebGL technique for highlighting objects without changing their appearance
- 2,034