-
Notifications
You must be signed in to change notification settings - Fork 16
Home
Hydro is a module for Godot 3.2 which allows rigid bodies to behave realistically in water. When an object enters the water, three forces will be calculated:
- Buoyancy - the pressure of the water causes an object to float.
- Drag - an object moving through the water will slow down. The amount of drag is dependent on the object's shape.
- Lift - an object moving through the water at an angle will generate lift, the way a boat lifts out of the water. This is also dependent on the object's shape.
In addition to applying these forces, this module also does the following:
- Allows you to plug in your own wave-generation function to have your objects bob in the ocean.
- Allows you to set a flow direction for features such as rivers.
- Contains convenience nodes for you to apply thrust and steering, making it easy to create a boat or other watercraft.
A simple demo project is included with the source code. You can also find more advanced examples in the Showcase.
For performance reasons, this is a native module that must be built from source with Godot.
- Get the source code for Godot: https://docs.godotengine.org/en/stable/development/compiling/getting_source.html
- Enter the source code directory, and go to the modules directory. Clone this project into a folder called "hydro":
git clone https://github.com/godot-extended-libraries/hydro.git
- Build Godot for your platform: https://docs.godotengine.org/en/stable/development/compiling/introduction_to_the_buildsystem.html
With this module, Godot will contain a several new Node types. HydroRigidBody will do everything a normal RigidBody does, and adds the hydrodynamic forces on top of it.
WaterArea defines the area in which hydrodynamic forces will be calculated.
To customize the ocean height or add waves of your WaterArea, attach a script to your WaterArea and add a new function:
func _get_water_heights(positions):
var ret : PoolVector3Array
for p in positions:
ret.append(Vector3(p.x, 42, p.z))
return ret
This example will set the water's global Y position to 42. You could also hook the Y value into another function that generates waves.
The HydroRigidBody will expect one mesh to be a direct child. This mesh will be the only one used to calculate the buoyancy and related forces. This mesh may be concave, but it must be closed and not intersect itself. Buoyancy is related to volume, and an open shape does not have a volume. If you have other display meshes that you want to come along for the ride, place them as a child of the main one. For example, you could place your boat hull as the main mesh, and then have your deck fittings, outboard motor, etc as child meshes. If you want a hollow boat, this can be done as well using some shader tricks.
For developing a pilotable watercraft, there are several additional convenience nodes:
- WatercraftBallast allows off-center weight to be added to a HydroRigidBody.
- WatercraftPropulsion provides directional thrust only when its node is underwater.
- WatercraftRudder parametrically generates a steerable rudder.
If you have a complex object that is not floating properly, you can add an ImmediateGeometry node as a child of the HydroRigidBody, and it will automatically be used for debugging. While your game is running, this will display a wireframe of the submerged portion of your hull, a line representing each of the forces that are generated, and a diamond shape representing the wave heights immediately surrounding your object. You can use these to diagnose whether something is out of place.
- Make sure your shape is totally closed.
- Make sure all your normals face outwards. If you see one force line facing the opposite direction of the rest, it's a good bet you need to check that face.
- Make sure you set a realistic mass. For reference, a 1m cube of styrofoam weighs 50 kg, so dropping in a default 2x2x2 cube and leaving the mass set at the default 1 kg will not work very well.
- If you want an object to start in the water, place it as close to its "rest" height on the water as possible. Starting a little too high is usually better than starting too low.
- Make sure that your water height function returns the desired height in global coordinates, and that your wave height does not exceed the bounds of its enclosing WaterArea.
Up to and including Godot 3.2.3, there is an engine bug in which getting the area of a face returns twice the area it should. This impacts both the buoyancy and the drag/lift forces that are generated. This module works around the bug, but if the problem is fixed on the engine side the results will become incorrect. Should you need to disable the workaround, you can switch the macro here.
The stresstest.gd
script can be used to generate and run a test in which one or more cubes will drop through a WaterArea.
You can configure the test using the following arguments:
--cubes=n
The number of cubes to generate
--density=n
Sets the density of the cubes. Must be larger than 1.0 in order to sink.
--depth=n
Exit when all cubes sink to this depth
--faces=n
The number of faces to generate per-cube. For exact results, use 6 times a square number.
For example, 3 * 3 * 6 = 54 faces.
--size=n
The size of each cube
After the test is complete, some statistics will be printed to the console. These tests may be useful for performance profiling, benchmarking, or regression testing.
Example Command:
bin/godot.x11.opt.tools.64.llvm -s modules/hydro/tests/stresstest.gd --cubes=1000 --faces=54 --size=1 --depth=5 --density 1.5
If you are interested in how the physics work, here are some resources I found useful or insightful along the way:
- http://www.iforce2d.net/b2dtut/buoyancy
- https://www.engineeringtoolbox.com/lift-drag-fluid-flow-d_1657.html
- https://www.gamasutra.com/view/news/237528/Water_interaction_model_for_boats_in_video_games.php
- https://www.gamasutra.com/view/news/263237/Water_interaction_model_for_boats_in_video_games_Part_2.php
While my approach has some differences, the writeup in the final two links above does an excellent job explaining and visualizing the physics involved.