2022-03-18 17:46:08 +01:00
|
|
|
.. _doc_csg_tools:
|
|
|
|
|
|
|
|
Prototyping levels with CSG
|
|
|
|
===========================
|
|
|
|
|
|
|
|
CSG stands for **Constructive Solid Geometry**, and is a tool to combine basic
|
|
|
|
shapes or custom meshes to create more complex shapes. In 3D modelling software,
|
|
|
|
CSG is mostly known as "Boolean Operators".
|
|
|
|
|
|
|
|
Level prototyping is one of the main uses of CSG in Godot. This technique allows
|
|
|
|
users to create simple versions of most common shapes by combining primitives.
|
|
|
|
Interior environments can be created by using inverted primitives.
|
|
|
|
|
|
|
|
.. note:: The CSG nodes in Godot are mainly intended for prototyping. There is
|
|
|
|
no built-in support for UV mapping or editing 3D polygons (though
|
|
|
|
extruded 2D polygons can be used with the CSGPolygon node).
|
|
|
|
|
|
|
|
If you're looking for an easy to use level design tool for a project,
|
|
|
|
you may want to use `Qodot <https://github.com/Shfty/qodot-plugin>`__
|
|
|
|
instead. It lets you design levels using
|
|
|
|
`TrenchBroom <https://kristianduske.com/trenchbroom/>`__ and import
|
|
|
|
them in Godot.
|
|
|
|
|
|
|
|
.. image:: img/csg.gif
|
|
|
|
|
|
|
|
Introduction to CSG nodes
|
|
|
|
-------------------------
|
|
|
|
|
|
|
|
Like other features of Godot, CSG is supported in the form of nodes. These are
|
|
|
|
the CSG nodes:
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
- `CSGBox`
|
|
|
|
- `CSGCylinder` (also supports cone)
|
|
|
|
- `CSGSphere`
|
|
|
|
- `CSGTorus`
|
|
|
|
- `CSGPolygon`
|
|
|
|
- `CSGMesh`
|
|
|
|
- `CSGCombiner`
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
.. image:: img/csg_nodes.png
|
|
|
|
|
|
|
|
.. image:: img/csg_mesh.png
|
|
|
|
|
|
|
|
CSG tools features
|
|
|
|
~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Every CSG node supports 3 kinds of boolean operations:
|
|
|
|
|
|
|
|
- **Union:** Geometry of both primitives is merged, intersecting geometry
|
|
|
|
is removed.
|
|
|
|
- **Intersection:** Only intersecting geometry remains, the rest is removed.
|
|
|
|
- **Subtraction:** The second shape is subtracted from the first, leaving a dent
|
|
|
|
with its shape.
|
|
|
|
|
|
|
|
.. image:: img/csg_operation_menu.png
|
|
|
|
|
|
|
|
.. image:: img/csg_operation.png
|
|
|
|
|
|
|
|
CSGPolygon
|
|
|
|
~~~~~~~~~~
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
The `CSGPolygon` node extrude along a Polygon drawn in
|
2022-03-18 17:46:08 +01:00
|
|
|
2D (in X, Y coordinates) in the following ways:
|
|
|
|
|
|
|
|
- **Depth:** Extruded back a given amount.
|
|
|
|
- **Spin:** Extruded while spinning around its origin.
|
|
|
|
- **Path:** Extruded along a Path node. This operation is commonly called
|
|
|
|
lofting.
|
|
|
|
|
|
|
|
.. image:: img/csg_poly_mode.png
|
|
|
|
|
|
|
|
.. image:: img/csg_poly.png
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
.. note:: The **Path** mode must be provided with a `Path`
|
2022-03-18 17:46:08 +01:00
|
|
|
node to work. In the Path node, draw the path and the polygon in
|
|
|
|
CSGPolygon will extrude along the given path.
|
|
|
|
|
|
|
|
|
|
|
|
Custom meshes
|
|
|
|
~~~~~~~~~~~~~
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
Any mesh can be used for `CSGMesh`; the mesh can be
|
2022-03-18 17:46:08 +01:00
|
|
|
modelled in other software and imported into Godot. Multiple materials are
|
|
|
|
supported. There are some restrictions for geometry:
|
|
|
|
|
|
|
|
- it must be closed,
|
|
|
|
- it must not self-intersect,
|
|
|
|
- it must not contain internal faces,
|
|
|
|
- every edge must connect to only two other faces.
|
|
|
|
|
|
|
|
.. image:: img/csg_custom_mesh.png
|
|
|
|
|
|
|
|
CSGCombiner
|
|
|
|
~~~~~~~~~~~
|
|
|
|
|
2023-01-12 19:30:47 +01:00
|
|
|
The `CSGCombiner` node is an empty shape used for
|
2022-03-18 17:46:08 +01:00
|
|
|
organization. It will only combine children nodes.
|
|
|
|
|
|
|
|
Processing order
|
|
|
|
~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Every CSG node will first process its children nodes and their operations:
|
|
|
|
union, intersection or subtraction, in tree order, and apply them to itself one
|
|
|
|
after the other.
|
|
|
|
|
|
|
|
.. note:: In the interest of performance, make sure CSG geometry remains
|
|
|
|
relatively simple, as complex meshes can take a while to process.
|
|
|
|
If adding objects together (such as table and room objects), create
|
|
|
|
them as separate CSG trees. Forcing too many objects in a single tree
|
|
|
|
will eventually start affecting performance.
|
|
|
|
Only use binary operations where you actually need them.
|
|
|
|
|
|
|
|
Prototyping a level
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
We will prototype a room to practice the use of CSG tools.
|
|
|
|
|
|
|
|
.. tip:: Working in **Orthogonal** projection gives a better view when combining
|
|
|
|
the CSG shapes.
|
|
|
|
|
|
|
|
Our level will contain these objects:
|
|
|
|
|
|
|
|
- a room,
|
|
|
|
- a bed,
|
|
|
|
- a lamp,
|
|
|
|
- a desk,
|
|
|
|
- a bookshelf.
|
|
|
|
|
|
|
|
Create a scene with a Spatial node as root node.
|
|
|
|
|
|
|
|
.. tip:: The default lighting of the environment doesn't provide clear shading
|
|
|
|
at some angles. Change the display mode using **Display Overdraw** in
|
|
|
|
the 3D viewport menu, or add a DirectionalLight node to help you see
|
|
|
|
clearly.
|
|
|
|
|
|
|
|
.. image:: img/csg_overdraw.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Create a CSGBox and name it `room`, enable **Invert Faces** and change the
|
2022-03-18 17:46:08 +01:00
|
|
|
dimensions of your room.
|
|
|
|
|
|
|
|
.. image:: img/csg_room.png
|
|
|
|
|
|
|
|
.. image:: img/csg_room_invert.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Next, create a CSGCombiner and name it `desk`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
A desk has one surface and 4 legs:
|
|
|
|
|
|
|
|
- Create 1 CSGBox children node in **Union** mode for the surface
|
|
|
|
and adjust the dimensions.
|
|
|
|
- Create 4 CSGBox children nodes in **Union** mode for the legs
|
|
|
|
and adjust the dimensions.
|
|
|
|
|
|
|
|
Adjust their placement to resemble a desk.
|
|
|
|
|
|
|
|
.. image:: img/csg_desk.png
|
|
|
|
|
|
|
|
.. note:: CSG nodes inside a CSGCombiner will only process their operation
|
|
|
|
within the combiner. Therefore, CSGCombiners are used to organize
|
|
|
|
CSG nodes.
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Create a CSGCombiner and name it `bed`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
Our bed consists of 3 parts: the bed, the mattress and a pillow. Create a CSGBox
|
|
|
|
and adjust its dimension for the bed. Create another CSGBox and adjust its
|
|
|
|
dimension for the mattress.
|
|
|
|
|
|
|
|
.. image:: img/csg_bed_mat.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
We will create another CSGCombiner named `pillow` as the child of `bed`.
|
2022-03-18 17:46:08 +01:00
|
|
|
The scene tree should look like this:
|
|
|
|
|
|
|
|
.. image:: img/csg_bed_tree.png
|
|
|
|
|
|
|
|
We will combine 3 CSGSphere nodes in **Union** mode to form a pillow. Scale the
|
|
|
|
Y axis of the spheres and enable **Smooth Faces**.
|
|
|
|
|
|
|
|
.. image:: img/csg_pillow_smooth.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Select the `pillow` node and switch the mode to **Subtraction**; the combined
|
2022-03-18 17:46:08 +01:00
|
|
|
spheres will cut a hole into the mattress.
|
|
|
|
|
|
|
|
.. image:: img/csg_pillow_hole.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Try to re-parent the `pillow` node to the root `Spatial` node; the hole will
|
2022-03-18 17:46:08 +01:00
|
|
|
disappear.
|
|
|
|
|
|
|
|
.. note:: This is to illustrate the effect of CSG processing order.
|
|
|
|
Since the root node is not a CSG node, the CSGCombiner nodes are
|
|
|
|
the end of the operations; this shows the use of CSGCombiner to
|
|
|
|
organize the CSG scene.
|
|
|
|
|
|
|
|
Undo the re-parent after observing the effect. The bed you've built should look
|
|
|
|
like this:
|
|
|
|
|
|
|
|
.. image:: img/csg_bed.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Create a CSGCombiner and name it `lamp`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
A lamp consists of 3 parts: the stand, the pole and the lampshade.
|
|
|
|
Create a CSGCylinder, enable the **Cone** option and make it the stand. Create
|
|
|
|
another CSGCylinder and adjust the dimensions to use it as a pole.
|
|
|
|
|
|
|
|
.. image:: img/csg_lamp_pole_stand.png
|
|
|
|
|
|
|
|
We will use a CSGPolygon for the lampshade. Use the **Spin** mode for the
|
|
|
|
CSGPolygon and draw a `trapezoid <https://en.wikipedia.org/wiki/Trapezoid>`_
|
|
|
|
while in **Front View** (numeric keypad 1); this shape will extrude around the
|
|
|
|
origin and form the lampshade.
|
|
|
|
|
|
|
|
.. image:: img/csg_lamp_spin.png
|
|
|
|
|
|
|
|
.. image:: img/csg_lamp_polygon.png
|
|
|
|
|
|
|
|
.. image:: img/csg_lamp_extrude.png
|
|
|
|
|
|
|
|
Adjust the placement of the 3 parts to make it look like a lamp.
|
|
|
|
|
|
|
|
.. image:: img/csg_lamp.png
|
|
|
|
|
2023-01-12 19:43:03 +01:00
|
|
|
Create a CSGCombiner and name it `bookshelf`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
We will use 3 CSGBox nodes for the bookshelf. Create a CSGBox and adjust its
|
|
|
|
dimensions; this will be the size of the bookshelf.
|
|
|
|
|
|
|
|
.. image:: img/csg_shelf_big.png
|
|
|
|
|
|
|
|
Duplicate the CSGBox and shorten the dimensions of each axis and change the mode
|
|
|
|
to **Subtraction**.
|
|
|
|
|
|
|
|
.. image:: img/csg_shelf_subtract.png
|
|
|
|
|
|
|
|
.. image:: img/csg_shelf_subtract_menu.png
|
|
|
|
|
|
|
|
You've almost built a shelf. Create one more CSGBox for dividing the shelf into
|
|
|
|
two levels.
|
|
|
|
|
|
|
|
.. image:: img/csg_shelf.png
|
|
|
|
|
|
|
|
Position your furniture in your room as you like and your scene should look
|
|
|
|
this:
|
|
|
|
|
|
|
|
.. image:: img/csg_room_result.png
|
|
|
|
|
|
|
|
You've successfully prototyped a room level with the CSG tools in Godot.
|
|
|
|
CSG tools can be used for designing all kinds of levels, such as a maze
|
|
|
|
or a city; explore its limitations when designing your game.
|
|
|
|
|
|
|
|
Using prototype textures
|
|
|
|
------------------------
|
|
|
|
|
2023-01-12 19:29:11 +01:00
|
|
|
Godot's `doc_spatial_material` supports *triplanar mapping*, which can be
|
2022-03-18 17:46:08 +01:00
|
|
|
used to automatically apply a texture to arbitrary objects without distortion.
|
|
|
|
This is handy when using CSG as Godot doesn't support editing UV maps on CSG
|
|
|
|
nodes yet. Triplanar mapping is relatively slow, which usually restricts its
|
|
|
|
usage to organic surfaces like terrain. Still, when prototyping, it can be used
|
|
|
|
to quickly apply textures to CSG-based levels.
|
|
|
|
|
|
|
|
.. note:: If you need some textures for prototyping, Kenney made a
|
|
|
|
`set of CC0-licensed prototype textures <https://kenney.nl/assets/prototype-textures>`__.
|
|
|
|
|
|
|
|
There are two ways to apply a material to a CSG node:
|
|
|
|
|
|
|
|
- Applying it to a CSGCombiner node as a material override
|
|
|
|
(**Geometry > Material Override** in the Inspector). This will affect its
|
|
|
|
children automatically, but will make it impossible to change the material in
|
|
|
|
individual children.
|
|
|
|
- Applying a material to individual nodes (**Material** in the Inspector). This
|
|
|
|
way, each CSG node can have its own appearance. Subtractive CSG nodes will
|
|
|
|
apply their material to the nodes they're "digging" into.
|
|
|
|
|
|
|
|
To apply triplanar mapping to a CSG node, select it, go to the Inspector, click
|
|
|
|
the **[empty]** text next to **Material Override** (or **Material** for
|
|
|
|
individual CSG nodes). Choose **New SpatialMaterial**. Click the newly created
|
|
|
|
material's icon to edit it. Unfold the **Albedo** section and load a texture
|
|
|
|
into the **Texture** property. Now, unfold the **Uv1** section and check
|
|
|
|
**Triplanar**. You can change the texture offset and scale on each axis by
|
|
|
|
playing with the **Scale** and **Offset** properties just above. Higher values
|
|
|
|
in the **Scale** property will cause the texture to repeat more often.
|
|
|
|
|
|
|
|
.. tip:: You can copy a SpatialMaterial to reuse it across CSG nodes. To do so,
|
|
|
|
click the dropdown arrow next to a material property in the Inspector
|
|
|
|
and choose **Copy**. To paste it, select the node you'd like to apply
|
|
|
|
the material onto, click the dropdown arrow next to its material
|
|
|
|
property then choose **Paste**.
|