Introduction
The VFX explained here is inspired by one of Keijiro’s works, which implements cool deforming character visual effects with Geometry Shader in Unity.
I also imitated those works in Unity before like below.
data:image/s3,"s3://crabby-images/9f850/9f8502848ba2e696ded9de175510f4c78c869c69" alt=""
data:image/s3,"s3://crabby-images/bde22/bde2265e757e3cd8b51cf5f76dc081901b54310f" alt=""
data:image/s3,"s3://crabby-images/e4ea6/e4ea606ad51512b5b13b8ab36d1a5d4828c8b6eb" alt=""
data:image/s3,"s3://crabby-images/2398a/2398a160d228915845f340cd4ec30640d3e4c6e0" alt=""
https://github.com/YoHana19/VoxelSizeAnimation
https://github.com/YoHana19/VoxelTeleporter
https://github.com/YoHana19/VoxelTeleporter2
https://github.com/YoHana19/Bandlizer
This time, I decided to reproduce them with Houdini and created the first three effects listed above.
アドベントカレンダー用①
— HeyYo (@yo_hanashima) December 17, 2021
Keijiro神の作例をHoudiniで再現#Houdini pic.twitter.com/gzYylIdSZz
アドベントカレンダー用②
— HeyYo (@yo_hanashima) December 17, 2021
Keijiro神の作例をHoudiniで再現#Houdini pic.twitter.com/o3X56orD9F
アドベントカレンダー用③
— HeyYo (@yo_hanashima) December 18, 2021
Keijiro神の作例をHoudiniで再現#Houdini pic.twitter.com/qDJeDYTudP
The actual working data is also available for purchase here, so feel free to check it out if you’re interested.
https://heyyohanashima.gumroad.com/l/svdbo
Overview
The concept behind the creation process for all the examples listed above is the same: it involves processing the triangular polygons that make up the geometry using a foreach
loop, transforming them into shapes like squares, cubes, etc., one by one.
At this stage, we make the size, center position, and deformation timing customizable for each primitive from the outside to create variations.
So, I’ll first explain the common transformation process for a single triangular polygon geometry before delving into each example.
Transformation Process From Single Polygon
data:image/s3,"s3://crabby-images/3ed10/3ed10224faaa85eb7b89fe63700ef8cb2876c438" alt=""
There are three types of transformations: triangle transformation, square transformation, and cube transformation. The triangle transformation only alters the size of the triangular shape. In this example, about 90% of the polygons simply shrink or enlarge and disappear or reappear. If all polygons were transformed into squares or cubes, the result would be a cluttered mess that’s hard to see.
These transformation processes are used in the actual examples, so I’ve encapsulated them into a subnetwork.
Triangle Transformation
data:image/s3,"s3://crabby-images/d5ad1/d5ad1f5b0088d34a8ab9cbf0fec6acf0dfcd908f" alt=""
data:image/s3,"s3://crabby-images/51013/51013b221e086a3a0a9aa917f556d24c73833207" alt=""
First, I check if the necessary attributes are present. The tri_morph
attribute controls the distance of each vertex from the center of the polygon, which in turn controls the size of the triangular polygon.
data:image/s3,"s3://crabby-images/b0c34/b0c34da4a4441244f1ebdfc320d57238fd625d24" alt=""
Using the “Extract Centroid” node, I store the centroid (the center of the polygon) in an attribute named centroid
.
data:image/s3,"s3://crabby-images/e3826/e3826b7d34cc00ad9dbab99c6f0c35dc6260069e" alt=""
Then, by multiplying the tri_morph
value by the vector from the centroid to each vertex, the size is controlled. When the tri_morph
value is 0, the polygon becomes a point, so it is deleted.
data:image/s3,"s3://crabby-images/96bd3/96bd3cc3aae6b62f07cf147c7b2c72ff1b63fab9" alt=""
Square Transformation
data:image/s3,"s3://crabby-images/43c27/43c2726553db7141d6c6171c619269a8dde031d6" alt=""
This transformation is slightly customized to match the example. It is not entirely generalized. Specifically, the square shape is always aligned with the Y-axis, and the pivot and center positions are used to control the orientation of the face. More details follow.
data:image/s3,"s3://crabby-images/be8d5/be8d54779ff8097250cbeccebbad38bb77f83dbe" alt=""
First, I check the required attributes. The attributes used for control are below.
data:image/s3,"s3://crabby-images/30bd7/30bd72d988564c95bc727af534405cc82b7b1da4" alt=""
quad_size: The size of the square
quad_pivot: The starting point of the direction vector for the face
quad_center] The position of the center of the square
Next, I add one point and transform the polygon into a square by adding the new point to the polygon’s vertices. The position and UV values can be chosen arbitrarily. For this example, I added the point at the position of the second vertex and used the UV of the first vertex for the new square’s UV.
data:image/s3,"s3://crabby-images/4690a/4690aaf2cc8c625b9e5be05ca9b01aad3b98338a" alt=""
For each vertex, I decide the direction (in tangent space) from the center and store it in tan_dir
. This can be chosen arbitrarily, but the key is ensuring that the vertices are ordered in a clockwise direction to maintain the correct orientation of the polygon.
data:image/s3,"s3://crabby-images/5331b/5331b67c105aa517c4b23453ece1652621bc7b40" alt=""
I then proceed with the transformation. First, I compute the direction from the center to each vertex in world space (converting the tangent XYZ directions to world space).
data:image/s3,"s3://crabby-images/8231d/8231d891fa8405471382712b1b5ed829d799bfd9" alt=""
The tangent Y direction (wld_tan_y
) is parallel to the Y-axis, so it is set to (0, 1, 0).
The tangent Z direction (wld_tan_z
) is determined by the vector from the pivot to the center (ignoring the Y-axis since it must remain parallel to the Y-axis). This ensures the square faces outward from the pivot.
The tangent X direction (wld_tan_x
) is calculated by taking the cross product of wld_tan_x
and wld_tan_z
.
data:image/s3,"s3://crabby-images/59145/59145e36d601cd12b23c2866d0c3fbaedb2182f7" alt=""
Using these results, I multiply the precomputed tan_dir
by the vectors, add the center position, and compute the position of each vertex in world space. The size of the square can be controlled by multiplying the result by the size
vector.
Finally, I interpolate between the original position and the square position to control the degree of transformation.
Afterward, I fuse any overlapping points to ensure clean geometry.
Cube Transformation
data:image/s3,"s3://crabby-images/11ee7/11ee755c9bcdb6e8a313867d1961292a8f33b0d2" alt=""
data:image/s3,"s3://crabby-images/87c17/87c172196d547ea770e7b67d81b8dfcb84141a93" alt=""
This process is a bit more complex, especially when it comes to matching up the newly created vertices with the original vertices.
First, I check the required attributes as same as others.
data:image/s3,"s3://crabby-images/9f4c5/9f4c5e042faedcc66bcf65cfcde66ca91c2295ae" alt=""
cube_morph: Controls the degree of transformation (0 = triangle, 1 = cube)
cube_offset: The offset from the center of the original polygon
These attributes are moved into detail attributes, and the average distance from the center to each vertex is also stored in detail attributes to reflect the original polygon size to scale the cube after transformation.
data:image/s3,"s3://crabby-images/b5e03/b5e036d9d31d1d5882ef46bb6c0073c2904d65ca" alt=""
data:image/s3,"s3://crabby-images/33c95/33c95c03b3d98b6484525d7ac93f02820e085948" alt=""
size
attribute).Next, I will add 8 points using the “Add” node, set their original positions (corresponding to one of the vertices of the original triangular polygon) as well as their deformed positions, and create polygons for each face of the cube.
There are a couple of important considerations:
- Vertex Order: The direction of the polygon’s face is determined by the clockwise order of the vertices, so I must be careful to maintain the correct order when creating the polygons.
- Mapping Original to New Vertices: Each cube face must map to the original triangular vertices in such a way that all three vertices of the original triangle are included in each cube face.
今回は、下の写真になるように各種設定しました。
data:image/s3,"s3://crabby-images/a1c3c/a1c3c189bd37c86b76fffa78608ed5d27489a935" alt=""
Red Number: Vertex Index of Original Polygon
I create a origin
attribute on the new vertices to store the positions of the original vertices. I then add the cube_offset
to the center position and multiply the cube_scale
by the size of the original polygon to determine the size of the cube.
data:image/s3,"s3://crabby-images/51306/51306b7b46533fc7e96da12428529f80720a4a0c" alt=""
The direction of each vertex from the center is calculated, scaled by the cube_scale
, and added to the center to determine the final position of each vertex in the cube, which is stored as. goal
attribute.
data:image/s3,"s3://crabby-images/30fe5/30fe5c794f1382320d2075c136c9c22bf5d96691" alt=""
Finally, I select the vertices to create the cube faces, ensuring that the vertices are ordered in a clockwise direction for proper polygon orientation.
data:image/s3,"s3://crabby-images/7e934/7e9347f933db7aab8b4b1eacb8f2ce91517bf582" alt=""
I also apply UV2 to each face of the cube to calculate emission mask later.
data:image/s3,"s3://crabby-images/62e23/62e230d2c5c15db5aad3518a9c7db10a86646b60" alt=""
To transfer the attributes from the original polygon, I will first move each vertex to the origin position, then use attribute transfer to transfer the attributes. (This step is optional.)
Finally, I will lerp the positions of the origin and goal using cube_morph
to control the deformation.
data:image/s3,"s3://crabby-images/6deb5/6deb53069b66e43bff633a51dfa0cc0de91d1233" alt=""
Note that the cube transformation process may not be perfectly smooth in this approach, so there could be a better way to achieve smoother results.
With the explanation of the common transformation process complete, I will now move on to explaining each example. The focus will be on how to animate the parameters that can be controlled externally in the deformation process to achieve the effects seen in the examples.
Example1: Quad Teleport
data:image/s3,"s3://crabby-images/46e8f/46e8f3642f28aa568dfb962400eb849bd88e0521" alt=""
I’ll start with some preparation.
data:image/s3,"s3://crabby-images/9b100/9b1006ca5b788aa871e4145733f52249ab357a1b" alt=""
First, I divide the rubbertoy
into triangular polygons.
data:image/s3,"s3://crabby-images/8b73a/8b73a68862e067d415c576601e69af127071079d" alt=""
Next, I create a morph
attribute, which will be used to control the degree of transformation during the deformation process.
This is done by animating two planes moving from top to bottom. When the y
position of a primitive is between the two planes, the morph
value changes from 0 to 1.
data:image/s3,"s3://crabby-images/9a92a/9a92a12bf0413b4a3bc373ca6fb26448bf75f0da" alt=""
Therefore, when the lower plane passes, the effect starts (morph=0
), and when the upper plane passes, it ends (morph=1
).
data:image/s3,"s3://crabby-images/30bd1/30bd1da299e38d5a305c863cee25d7f741423f62" alt=""
The set_group
node groups the triangles into two groups: one that shrinks and disappears and one that transforms into squares. Around 96% of the polygons will shrink and disappear.
data:image/s3,"s3://crabby-images/dd154/dd1546bbb23b796d4b70e914da5b28a977cea500" alt=""
I’ll store the primnum
as an id
for convenience when applying colors or other attributes later.
Now, let’s move on to explaining the control of the deformation parameters.
The triangle transformation is simple. I subtract the morph
value from 1 and assign it to the tri_morph
attribute to control the transformation for each primitive.
data:image/s3,"s3://crabby-images/1170f/1170f03aef8a480521d2217d007e799903ae629e" alt=""
data:image/s3,"s3://crabby-images/fcf3f/fcf3f468d6671e47f6cf57aec2f8fff942c58702" alt=""
data:image/s3,"s3://crabby-images/0cac2/0cac2a5b71ff51c41cd2d831f7482c227d655a5b" alt=""
Next, for the square deformation, I’ve also included an effect at the beginning where the triangular polygon is expanded, so triangle deformation is used a little as well. I’ll explain everything step by step, including that.
data:image/s3,"s3://crabby-images/650c6/650c63913d28ded2a5cccc9282cc8800e3b7531a" alt=""
data:image/s3,"s3://crabby-images/ac6b8/ac6b84399d3c2dddeda0e5559c9a2ae52e910724" alt=""
First, let’s control the quad_pivot
. In this example, quad_pivot
is used as the center point around which the square rotates. Since the rubbertoy model has a significantly different center point on the XZ plane for its top part, I use a plane animated for the morph effect to cut the rubbertoy, then calculate the center point for each frame and connect them into a line.
data:image/s3,"s3://crabby-images/d36ac/d36acc00fe9521d8125d667ad3317d442b86fc5d" alt=""
For this line, I perform a nearpoint
operation on each primitive to determine the closest point’s position, which is then assigned to the quad_pivot
.
data:image/s3,"s3://crabby-images/5979f/5979fe2d83177d7c76473606b898f31dc31cc223" alt=""
Next, to scale up the triangular polygon at the beginning, I set the value of tri_morph
using inflation_morph
. By controlling the ramp value, I create the initial expansion effect.
data:image/s3,"s3://crabby-images/d9024/d9024fea8aa0697448f19a2c8e90e30f183c9e32" alt=""
data:image/s3,"s3://crabby-images/dfb0f/dfb0fa9454b1c1833574444949a28d92230f30e3" alt=""
Then, using set_quad_center
, I define the position of the square’s center, quad_center
. I calculate vectors from the quad_pivot
position to each primitive, and use the ramp-controlled morph
value to rotate them. Additionally, I use the ramp-controlled morph
value to control the distance from the quad_pivot
, which animates quad_center
moving away from the pivot while rotating in sync with the morph effect.
data:image/s3,"s3://crabby-images/601c8/601c85b932794e1b3a6d743b52b35e1418be3e08" alt=""
data:image/s3,"s3://crabby-images/96ef7/96ef7506d17109dc02d01c1f8b48cdef62f5ed71" alt=""
For quad_size
, I control the ramp so that it ultimately reduces to 0, creating an animation where it disappears at the end.
data:image/s3,"s3://crabby-images/c7eac/c7eacf82023e061d133117f7ec304362d89c51ee" alt=""
data:image/s3,"s3://crabby-images/222a4/222a4cbcd388a2d8922d24126b4cee3c2ecbc58a" alt=""
Finally, I control the value of quad_morph
using the original morph
value and the ramp, adjusting it so that the square transformation is completed around the middle of the animation.
data:image/s3,"s3://crabby-images/ec5d6/ec5d6b42c875a1c05a8bacf3577f65ae1eec9efa" alt=""
data:image/s3,"s3://crabby-images/a56c1/a56c1529bca46c1560c92aeff7af5651118252a0" alt=""
With all the parameters set, I will now apply the triangle and square deformation processes to each primitive.
Since the deformation process can be heavy, I will cache it, and then adjust the look by setting necessary parameters like Color and Emission to match the morph values before finishing.
data:image/s3,"s3://crabby-images/d7ebb/d7ebbb54f3c455677f90a01885018551537fa070" alt=""
Example2: Voxel Animation
data:image/s3,"s3://crabby-images/21ece/21eceb6f36be3d76ce33284063887ce0cdc7063b" alt=""
This example is quite simple, and since the control of the morph
and the separation of triangle and cube deformations are almost the same as in Example 1, I will skip those details.
The key part is how to animate the cube_scale
, so I will explain that. It’s not very complex — I use the UV values of three vertices as seeds, animate each axis (x, y, z) with sine functions, multiply them by an appropriate scale, and then set the result to cube_scale
.
data:image/s3,"s3://crabby-images/06ca0/06ca05b5f075ba2e0ab9a30d20c80c4888ac0715" alt=""
After that, I apply the cube deformation process to each primitive.
Example3: Voxel Teleport
data:image/s3,"s3://crabby-images/8a561/8a5612ad4e2d918c1e59058e81eb87e54d60852b" alt=""
This example is also similar to Example 1 in terms of the base morph control and the separation of triangle and cube deformations, so I will skip those details.
The crucial part is controlling the animation of cube_scale
and cube_offset
, so I will explain that.
data:image/s3,"s3://crabby-images/85b04/85b04115595f4bc57d6cf98ead77b1fd20712f1c" alt=""
First, in cube_teleport_offset_scale
, I set the cube_scale
and cube_offset
. The animation starts with a tall cube falling from the top and gradually transforming into a regular cube, so I animate the values accordingly.
data:image/s3,"s3://crabby-images/22478/22478a1c66a2f697c02d81a5a4b27a1b1cc2bed9" alt=""
In this case, since the effect is to revert from the deformed state to the original, the morph
value changes from 1 to 0. The move
value, which controls the size of the offset and scale, changes from 1 to 0 as morph
transitions from 1 to 0.75. This means that during the first quarter of the animation, the cube falls back to its original position.
For the scale
, the y
value animates from a larger value to 1, while the xz
values animate from 0 to 1, giving the effect of a tall cube gradually becoming a regular cube.
I also introduce some noise to add variation to the animation.
Next, I want to animate the size slightly during the brief period before the falling cube transforms back into a triangle, similar to Example 2. I do this by adjusting the cube_scale
with cube_scale_anim
.
data:image/s3,"s3://crabby-images/6f652/6f6520ad66c717121b55b5f4070cb4c615071c5e" alt=""
Since the implementation is quite similar to Example 2, I’ll skip the details, but essentially, after the cube finishes falling, I control the timing of the animation with a ramp
.
data:image/s3,"s3://crabby-images/0d5b7/0d5b729656eec6d5f3a945e508db36f040b0c800" alt=""
I also control the timing of the deformation using the ramp
, ensuring that the cube smoothly transitions back into the original triangle at the right moment.
data:image/s3,"s3://crabby-images/619b5/619b5702d457808a59e1c76056502c36a767b576" alt=""
data:image/s3,"s3://crabby-images/3de16/3de16a8eb3d3cc2ac6d1659312584f263af9be6f" alt=""
Finally, I apply the triangle and cube deformation processes to each primitive.
data:image/s3,"s3://crabby-images/a3788/a3788e3070a2f14e91c43d02587ecbf1ddaf31ea" alt=""
Visual Adjustments
I won’t go into detailed explanations regarding the visual adjustments, so if you’re interested, please check the data I provide below.
https://heyyohanashima.gumroad.com/l/svdbo
Here, I’ll just provide a brief overview.
I directly edited the materials inside the rubbertoy, adding only the emission values. Essentially, I used uv2
and other attributes to create emission effects around the edges of the square and cube faces. For the voxel teleport, I set the entire surface to emit when the object first falls.
The control of the emission values is also handled by adjusting the morph
value using a ramp.
For the glow effect, I applied it in post-processing using Houdini’s compositing tools. Since I didn’t know a good way to extract only the emission areas, I used a LumaMatte to isolate the high-luminance areas, applied a blur, and added the glow effect that way.
Conclusion
My first impression is that it’s “heavy!” haha.
If I had implemented this in Unity using Geometry Shader, it would have run smoothly in real-time, but it took about 15 minutes just to cache 180 frames…
However, one thing I like about Houdini is that it works exactly as written. When you implement it in Geometry Shader, there are often parts that become black boxes and aren’t as clear. In contrast, Houdini follows the instructions precisely, which makes implementation much easier.
Ultimately, the key is how you animate the parameters. For this example, I simply borrowed the ideas that Keijiro-sensei had already developed, so moving forward, I’d like to see what variations I can come up with on my own. Whenever I feel like it, I’ll challenge myself to explore new possibilities.