Stylized Grass Shader Documentation

Stylized Grass Shader

Stylized Grass Shader
Version: 1.3.0
Review

4.Placing grass #

There are several methods to populate a world with grass in Unity, this section outlines them and their (dis)advantages.

This asset is a grass shader, hence does not contain any tools to achieve this. Such tools and systems should be considered entirely separate, since they’re often quite complex in their own right. The reality today is that Unity does not have the proper tools needed for AAA vegetation rendering.

While the shader has been written to be as efficient as possible, the method used for rendering grass meshes will greatly determine how and where performance will be affected. In any event, I strongly recommend to learn and use the Profiler and Frame Debugger tools to get an insight into how rendering is performed in your project. This is the only correct way to gauge where rendering optimization is needed.

In the Assets/Stylized Grass Shader/Prefabs folder you’ll find several pre-made prefabs, ready to use with any of the techniques described below.

Unity terrain vegetation system

This is arguably the most accessible method, and does not require any third-party tools.

Unity 2021.1 and older

It is important to understand that the grass/detail object system does not support meshes with custom shaders. It’s designed to use the built-in grass shader. To get around this, add a grass prefab as a “tree” instead. This way you can paint custom grass using the terrain tools.

The Unity vegetation system won’t deliver high performance when grass (as trees) is placed in large amounts. Using this method, grass has to be placed sparingly, regardless if using this grass shader or not.

If you’re using navmeshes then this method is not advised, since trees placed through the terrain system are automatically marked as navmesh obstacles. Meaning that navmesh agents won’t able to navigate through grass (placed as trees). Though, using Unity’s “AI Navigation” system instead doesn’t have this limitation, it effectively offers an improved way of baking navmeshes (as volumes, no longer tied to a single scene).

Unity 2021.2+

The terrain system now supports using custom meshes/material for grass. Except LODs aren’t supported.

Because of this, you’re required to create a grass prefab that just uses a single Mesh Renderer. To do so, open the grass prefab of your choice, and drag the “_LOD0” object into the Project window to create a new prefab.

In the Prefabs/TerrainFlowers & TerrainGrass you’ll find prefabs that have been altered in this way. Only these can be used with the terrain! (see Unity manual)

Pros:

Cons:

  • Grass can’t align to terrain surface and will stick out on hills/slopes.
  • No options for random scale, rotation or density
  • Grass can’t cast shadows
  • Not optimized for dense grass or large amounts (Unity 2021.1 and older)

Placing grass as GameObjects

By dragging a grass prefab into the scene you’ll have full control over it, just like any other object. It goes without saying that placing a piece grass one by one is not an option. For this reason you can use any prefab painting tool, such as PolyBrush.

Note that populating a scene with a large amount of individual GameObjects will increase CPU load, scene file size, culling overhead, and loading times. So the amount of grass you can place is limited by how much performance legroom your project already has in these area. The shader uses GPU Instancing, so draw calls in any case will fortunately be limited: one for depth, lighting and one per shadow cascade.

This method is definitely not recommended for large terrains, spanning open fields of dense grass.

Pros:

  • Prefab painters usually apply random scale/rotations.
  • Easy to manage to like any other GameObject
  • Material changes are directly visible
  • Meshes can be used with Occlusion Culling

Cons:

  • Has to be manually placed or painted
  • Requires CPU resources for culling and batching the individual Mesh Renderers (per camera)

Nature Renderer

Nature Renderer uses the terrain’s tree and details data, but renders everything using its own system. It allows you to use the terrain vegetation system as you would normally, but get all the modern rendering methods required for high amounts of vegetation (GPU instancing, cell-based and frustrum culling). As a result, this would work with other systems that spawn grass on terrain such as Gaia, Map Magic or Vegetation Spawner.

The grass shader is compatible with its “Procedural Instancing” technology. See the third party integrations page for more details.

Pros:

  • No change in workflow (grass prefabs can be used with the terrain detail system)
  • Renders large amounts of grass at relatively fast speed
  • Material changes are directly visible
  • Aligns grass to terrain surface
  • Stylized Grass supports its wind controller and “Procedural Instancing” system.

Cons:

  • Only renders on terrains

Vegetation Studio + Pro

Vegetation Studio is a system that utilizes GPU Instancing to draw identical meshes in only a few draw calls. It also performs frustum culling, to only render what’s visible in the camera field. It does not use any GameObjects, the grass is purely stored in memory and rendered through a custom render loop. Additionally, it provides a powerful rule-based system for spawning grass.

The grass shader is compatible with its Instanced Indirect rendering, which enables more than 1023 items to render in 1 draw call, next to GPU-accelerated culling.

Pros:

  • Renders large amounts of grass at relatively fast speed
  • Procedural rule-based placement of grass (in addition granular manual control)
  • Fast control over density and scale/rotation/alignment
  • Support for placement on meshes
  • Integrated material controller when selecting a grass item
  • LOD Crossfading

GPU Instancer

As the name imply, GPU Instancer is a system is a rendering framework that makes use of Unity’s instanced rendering API. It supports both the conversion of GameObjects and terrain vegetation data.

The grass shader is compatible with its Instanced Indirect rendering, which can be enabled at the bottom of the grass material UI.

[No pro’s/con’s are known, as I personally have no production experience using this tool]

    Tips and tricks

    • Individual grass objects should be rotated randomly on the Y axis between 0-180º. This greatly reduces visual repetition.
    • Scaling the objects randomly between 80-120% is also advised for the same reason.
    • Align the grass to the surface it is placed on. This will help with translating lighting, especially on hills.
    • Disabling shadow casting for the grass will greatly reduce draw calls, consider going in that direction as a style choice. Contact/Raytraced Shadows are an excellent alternative, but currently aren’t available in URP.
    • Unity’s SSAO will always look blurry on small objects such as grass, which is inherent to the technique. I recommend to use the Horizon Based Ambient Occlusion asset for far better results.
    Yes No Suggest edit
    Last updated on October 10, 2022
    40 of 40 users found this section helpful
    Suggest Edit