faking_global_illumination.rst 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. .. _doc_faking_global_illumination:
  2. Faking global illumination
  3. ==========================
  4. Why fake global illumination?
  5. -----------------------------
  6. Godot provides several global illumination (GI) techniques, all with their advantages
  7. and drawbacks. Nonetheless, it remains possible to avoid using any GI technique
  8. and use a handmade approach instead. There are a few reasons for using a
  9. "handmade" approach to global illumination instead of VoxelGI, SDFGI or
  10. baked lightmaps:
  11. - You need to have good rendering performance, but can't afford going through
  12. a potentially cumbersome lightmap baking process.
  13. - You need an approach to GI that is fully real-time *and* works in procedurally
  14. generated levels.
  15. - You need an approach to GI that is fully real-time *and* does not suffer from
  16. significant light leaks.
  17. The approaches described below only cover indirect diffuse lighting, not
  18. specular lighting. For specular lighting, consider using ReflectionProbes which
  19. are usually cheap enough to be used in conjunction with this fake GI approach.
  20. .. seealso::
  21. Not sure if faking global illumination with lights is suited to your needs?
  22. See :ref:`doc_introduction_to_global_illumination_comparison` for a
  23. comparison of GI techniques available in Godot 4.
  24. Faking DirectionalLight3D global illumination
  25. ---------------------------------------------
  26. While the sky provides its own directional lighting, the scene's main DirectionalLight3D
  27. node typically emits a large amount of light. When using a GI technique, this light
  28. would be reflected on solid surfaces and would bounce back on most outdoors shaded surfaces.
  29. We can fake this by adding a second DirectionalLight3D node with the following changes:
  30. - Rotate the light by 180 degrees. This allows it to represent lighting bounced
  31. by the main DirectionalLight3D node.
  32. - Set **Shadows** to **Off**. This reduces the secondary light's performance burden
  33. while also allowing shaded areas to receive *some* lighting (which is what we want here).
  34. - Set **Energy** to 10-40% of the original value. There is no "perfect" value,
  35. so experiment with various energy values depending on the light and your typical
  36. material colors.
  37. - Set **Specular** to ``0.0``. Indirect lighting shouldn't emit visible specular
  38. lobes, so we need to disable specular lighting entirely for the secondary light.
  39. .. note::
  40. This approach works best in scenes that are mostly outdoors. When going indoors,
  41. the secondary DirectionalLight3D's light will still be visible as this light
  42. has shadows disabled.
  43. This can be worked around by smoothly decreasing the secondary DirectionalLight3D's
  44. energy when entering an indoor area (and doing the opposite when leaving the indoor area).
  45. For instance, this can be achieved using an Area3D node and AnimationPlayer.
  46. Faking positional light global illumination
  47. -------------------------------------------
  48. It's possible to follow the same approach as DirectionalLight3D for positional
  49. lights (OmniLight3D and SpotLight3D). However, this will require more manual
  50. work as this operation needs to be repeated for every positional light node in
  51. the scene to look good.
  52. In an ideal scenario, additional OmniLight3Ds should be added at every location
  53. where a significant amount of light hits a bright enough surface. However, due
  54. to time constraints, this isn't always easily feasible (especially when
  55. performing procedural level generation).
  56. If you're in a hurry, you can place a secondary OmniLight3D node at the same position
  57. as the main OmniLight3D node.
  58. You can add this node as a child of the main OmniLight3D node to make it easy to
  59. move and hide both nodes at the same time.
  60. In the secondary OmniLight3D node, perform the following changes:
  61. - Increase the light's **Range** by 25-50%. This allows the secondary light to lighten
  62. what was previously not lit by the original light.
  63. - Set **Shadows** to **Off**. This reduces the secondary light's performance burden
  64. while also allowing shaded areas to receive *some* lighting (which is what we want here).
  65. - Set **Energy** to 10-40% of the original value. There is no "perfect" value,
  66. so experiment with various energy values depending on the light and its surroundings.
  67. - Set **Specular** to 0. Indirect lighting shouldn't emit visible specular lobes,
  68. so we need to disable specular lighting entirely for the secondary light.
  69. For SpotLight3D, the same trick can be used. In this case, the secondary OmniLight3D
  70. should be placed in a way that reflects where *most* light will be bounced.
  71. This is usually close to the SpotLight3D's primary impact location.
  72. In the example below, a SpotLight3D node is used to light up the room's floor.
  73. However, since there is no indirect lighting, the rest of the room remains
  74. entirely dark. In real life, the room's walls and ceiling would be lit up by
  75. light bouncing around. Using an OmniLight3D node positioned between the
  76. SpotLight3D's origin and the floor allows simulating this effect:
  77. .. image:: img/faking_global_illumination_comparison.webp