# Shader: rendinst_layered ## Overview This shader is used for objects where we want to apply multiple material textures (e.g., brick and stucco tiles that are stored in different textures). It can work with both tiles and unique textures. However, its primary purpose is for using tiles – it allows objects to be created without visible tiling, without the need for unique textures. The shader uses the following types of textures: - **UV1**: Global normal map – the normal map for the entire object. - **UV3**: Details (detail textures) with normal maps and speculars – 2 tiling material textures. - **UV2**: Overlay map – applies effects of leaks/wear and tear to the unwrapped parts of objects, over the detail textures (tiles). - **UV2**: Exclusion mask – globally indicates where certain materials should appear on the object. - **UV2**: Blending mask – allows to randomize the blending of detail textures with the exclusion mask across the model, resulting in diverse appearances of the object using the same textures. Let's take a closer look. ## Textures Used ### Texture Assignment For more details, see [here](./rendinst_layered.md#texture-assignment-1). ### Global Normal Map This is straightforward. For most simple objects (fences, bollards, sometimes houses), a standard normal map from our libraries is used. For more complex objects, unique normal maps are created. Their main purpose is to smooth out edges and chamfers. Since the objects themselves will be covered with material tiles (which contain their own normal maps), it's important to achieve smoothing on the global normal map for those aspects that the tile normal maps cannot smooth out. ```{important} Do not create the texture details of the object's materials on this texture. Only smooth out seams, worn edges, etc. ```

Without a global normal map

With a normal map showing chamfers

The global normal map uses **UV1**. No other texture uses this UV channel. ### Detail Textures (Details) Let's illustrate the concept using a brick house covered with stucco. **There are always two details!** - **Detail1** represents the material the object is made of. For instance, if the house wall is made of bricks, then Detail1 would be bricks. - **Detail2** represents the material covering the object (such as moss on a tree or dirt on a cobblestone). For the house wall covered in stucco, Detail2 would be the stucco. In this example, we use `modern_brick_brown_dirt_d.tif` and `stucco_new_d.tif`, along with their respective normal maps, which contain corresponding specular maps in their alpha channels.

1st map - Dirty Brickwork

2nd map - Fresh Stucco

In general, the process is straightforward, with one exception – we need to blend these textures automatically on the model. To achieve this, we use the alpha channel of the diffuse maps, which contains heightmaps: the lighter the pixel, the "heavier" it is. The material with the higher heightmap at any given point will dominate in that texel. By blending these two heightmaps in the shader, we achieve a non-uniform texture appearance: in some areas, the brickwork is visible because its heightmap is higher, while in others, the stucco dominates. For example:
What happens if we modify these heightmaps? Suppose we want the stucco to cover the entire brick surface completely. To do this, we need to eliminate all the light pixels in the brick's heightmap, as these are the areas where the brick shows through the stucco.

Making the heightmap entirely black:

Results in:

```{important} Of course, this approach isn't advisable – this was merely a demonstration of how heightmaps work. These maps are essential for creating a seamless, non-repeating overlay of stucco on top of the bricks. ``` **Details use UV3 mapping**, typically mapped using a 3x3x3 meter box without packing. ```{important} Ensure that the geometry does not exceed 7 tiles in the UV layout; otherwise, artifacts may appear on consoles. ``` ### Overlay Map The texture is named "overlay" because the shader applies it to the object using a method similar to the "overlay" blend mode in Photoshop – `multiply2X`. The purpose of the overlay is to give the object a "lived-in" appearance: stains, cracks, dirt, soot, etc.

Here is our house without the overlay (technically with a 50% gray overlay – completely disabling it would prevent the shader from functioning). We can see only the brick and stucco – the object looks plasticky:

And here's the house with the overlay applied:

### Universal Overlay For this demonstration, we used a universal overlay – `buildings_overlay_tex_d.tif`, which we apply to our houses and simpler structures like fences. It looks like this:
Conceptually, it's divided into 16 cells (with indices starting from 0, not 1):
- **Cells 0-1** are used to decorate the building's interior materials (walls with wallpaper/stucco). ```{note} As shown, these two cells contain four smaller cells each – ensure that the seams between these cells do not fall on a flat surface, or else a distinct vertical line will appear on the model! For example: ```
- **Cell 2** is used to decorate wooden roof trim, similar to this:
It should be mapped as follows: - Either using a horizontal plane of 55x55 meters. Vertical supports should be mapped separately in the vertical plane; - Or using box and flatten mapping. - **Cells 3, 6, 7** are currently unused. - **Cells 4-5** are used to decorate roofs: - **Cell 4** is for sloped roofs; - **Cell 5** is for flat roofs. - **Cells 8-15** are used to decorate the building's exterior. In the example above, **cell 11** was used. Here's how it works:
```{important} All external surfaces of the building, except for the roof, should only be mapped to **cell 8**. The shader will automatically shift the mapping to cells **9-15**. More on this in the shader settings. ``` ```{note} In the early stages of *War Thunder*'s development, universal overlays were widely used. However, as the project evolved, we transitioned to unique overlays. Consequently, the content of this section is now outdated. Despite this, universal overlays can still be employed, though they are better suited for smaller objects rather than larger structures like buildings. ``` ### Unique Overlay The universal overlay was designed to automate visual variety in objects – by creating one house, you can give it up to eight different appearances without any additional work. However, this map is not suitable for unique objects, as it cannot emphasize the distinctive features that make an object more "alive". Let's look at an example of a unique overlay:
As you can see, the entire texture is dedicated to a unique UV layout, reflecting soot marks on walls at bullet impact sites, rotting wood (small details), moss-covered foundations, etc. It was created for a partially ruined church:
### Exclusion Mask This texture resides in the alpha channel of the overlay map and therefore uses the same UV2. Let's examine it using the universal overlay's alpha channel as an example.
This is a black-and-white mask. It defines the global areas on the model where we want to preserve only the first detail or only the second. - **Black pixels** – only the first detail will remain. - **White pixels** – only the second detail will remain. - **Intermediate grayscale values** – adjust the global heightmaps of the two details. Why is this necessary? Let's see how the details mix with a 50% gray exclusion mask. (For clarity, we'll set the overlay to the same 50% gray so it doesn't interfere with the detail mixing.)
Several issues are apparent: 1. The tiling pattern is fully readable. Even after applying the overlay, this might not be corrected, as the overlay cells (when using the universal overlay) have varying levels of "dirtiness." For example, using **cell 8** produces the following result:
It's clear that the tiling is still visible. 2. Detail2 (stucco) blends with Detail1 (bricks) uniformly and boringly across the entire wall area. This can be somewhat improved by using a dirtier overlay. For example, let's try **cell 11**:
Yes, it looks better. However, problem 1 remains – the cells are used automatically, and the mapping will inevitably shift to a less detailed cell. The exclusion mask is designed to address these specific issues. It globally overrides the heightmaps of the details across the entire object mapped to it, causing the blending to be rough and uneven. This helps hide the tiling (it's hard to notice when Detail2 covers different parts of Detail1) and eliminates uniform blending. Let's switch the exclusion mask from gray to the original.
The result is clear – both issues are resolved. However, **cell 8** is rather sparse for both the overlay and the exclusion mask.
Let's see what happens if we shift the UV2 mapping to **cell 11**. In the screenshots above, it worked with a gray exclusion mask. Now let's see its full effect.

There is no visible tiling. The details blend randomly. The overlay adds even more detail. The more varied the exclusion mask, the smoother the detail blending. The more gradients and fewer sharp transitions it contains, the smoother the boundaries between details. ```{important} In summary, the exclusion mask controls where Detail1 (the object's base material) is visible – where its pixels are black–and where Detail2 (the object's covering) appears – where its pixels are white. Where the pixels are gray, the details blend based on their heightmaps, taking into account the "grayness" of the exclusion mask: - If the mask is 50% gray, only the heightmaps are used. - If it is darker than 50%, the heightmap of Detail1 is stronger. - If it is lighter than 50%, the heightmap of Detail2 takes precedence. ``` ### Blending Mask #### How It Works The blending mask is the final texture we utilize. Let's examine it using the example of the `brick_stucco_m.tif` texture, which was applied to the house we discussed earlier.
**What does it do?** Essentially, it performs the same function as the exclusion mask: - Black areas reveal Detail1. - White areas reveal Detail2. - Gray areas blend the two details based on their heightmaps and the gray value of the blending mask. Let's recall how the house looked without a blending mask (in the earlier screenshots, it was filled with 50% gray).
Now, let's see how the house appears with the `brick_stucco_m.tif` blending mask applied.
The difference is subtle – the exclusion mask creates a more noticeable change. This is because, on UV2, our house occupies a single cell, which is quite insignificant in the context of the blending mask.
So, why do we need a blending mask that seemingly duplicates the exclusion mask, but with less impact? There are four key points to understand: 1. In reality, details are not blended solely by the exclusion mask. 2. Similarly, details are not blended solely by the blending mask. 3. The exclusion mask and blending mask are combined to create the final global blending mask, which dictates how the details are blended. ```{note} These masks are combined using standard multiplication (not *Photoshop*'s Multiply mode). ``` 4. Our `rendinst_layered` shader randomly offsets the blending mask (the local one we're discussing) based on the position of the house on the map. To summarize in simpler terms: 1. Place the house on the map, for instance, at the center. 2. The blending mask and exclusion mask combine to form the global blending mask. - Black areas reveal Detail1. - White areas reveal Detail2. - Intermediate gray values influence the blending of details based on their heightmaps. 3. Move the house slightly, and the blending mask and exclusion mask are multiplied again, creating a new global blending mask and altering how the details blend. Here's how it looks in practice: ```{eval-rst} .. video:: _images/rendinst_layered_31.webm :width: 80% ``` Each movement triggers a reassembly of the global blending mask via multiplying the exclusion mask and the local blending mask, which results in the same object having a different appearance as it is repositioned. This is the core function of the blending mask. We cannot offset the exclusion mask since it contains a unique mapping of detail blending (even within the cells). The blending mask is designed precisely for this purpose. #### Specifics of Using the Blending Mask We used the `brick_stucco_m.tif` as our example. The name itself indicates that it represents stucco over brick, meaning it's intended for a specific type of detail blending – Detail2 (stucco) is layered over Detail1 (brick) with clear boundaries. Let's look at a real brick house with stucco, such as this one:
The concept becomes clear – stucco peels off in sheets, revealing the underlying wall material. How did we achieve this effect on our house? The current blending mask has sharp transitions between black and white, with almost no gradients. This means we either see the brick (black areas) or the stucco (white areas). But what if we want to create a stone wall with moss growing between the stones? We'd use a stone texture (`medieval_brick_d.tif`) and a moss texture (`moss_different_d.tif`). Let's examine these textures and their heightmaps. 1. **Stone Masonry**: Its heightmap clearly indicates that any Detail2 should only appear between the stones. Recall that the heightmap function – the lighter the area, the stronger the effect.
2. **Moss**: Its heightmap suggests that it doesn't matter where it grows. It's uniformly gray, meaning it will be overpowered by almost any other detail's heightmap.
Let's see how these details blend with the current blending mask. Remember, the moss heightmap will likely be overridden by the stone heightmap, so the moss should only appear between the stones – exactly what we want. However, what we get is moss applied over the entire wall surface.
**Why did this happen?** Because our blending mask is unsuitable. As stated earlier: - "What does the blending mask do? It functions exactly like the exclusion mask. Black areas reveal Detail1, white areas reveal Detail2, and gray areas blend the two details according to their heightmaps and the gray value of the blending mask." - "The current blending mask has sharp transitions between black and white, with almost no gradients. This means we either see the brick (black) or the stucco (white)." Our `brick_stucco_m.tif` blending mask has overridden the detail heightmaps and distributed them as it saw fit. **What to do in such cases?** Change the blending mask. The approach is straightforward: - Ensure that the detail heightmaps are not overridden. - Detail heightmaps are not overridden by gray. - We need smooth transitions between the details. **Conclusion**: We need a blending mask with lots of gray and few sharp boundaries – everything should be gradient-based. Let's try `gypsum_m.tif`. It's not a perfect mask, but it's better than the previous one. We can see that it contains a lot of gray, with minimal black/white areas and many gradients. There are some patches, but they're small and sparse.
Now, let's see how it performs.
The moss remains in some places, but mostly it has receded to the gaps between the stones. The remaining local patches can be removed if needed, but that would require additional methods. Remember that these masks will be multiplied, and the final appearance will continue to evolve: ```{eval-rst} .. video:: _images/rendinst_layered_38.webm :width: 80% ``` ### Conclusion A concise summary of how the shader uses textures: 1. Two details are blended based on their heightmaps (through alpha test). 2. The exclusion and blending masks combine to create the final global blending mask for detail blending (using standard multiplication, not *Photoshop*'s Multiply mode). This raises a question: Why not blend the details directly using this mask? Why do we need detail heightmaps as well? Because the blending mask has a large texel size, blending directly would result in blurry textures. Using the detail heightmaps allows for a sharper blend. 3. When the heightmaps are combined with the blended masks, the result is that within the large texel of the global blending mask (which covers the entire building), several smaller texels of the local blending mask are nested. These smaller texels are where the detail heightmaps interact. This process ensures that the image remains sharp, as the local blending mask allows for finer detail within each texel of the global mask. 4. Next, a normal map and an overlay map are applied on top of the blended details. These maps add texture and additional detail to the surface, enhancing the visual complexity of the material. 5. Finally, when the object using this shader is repositioned, the blending mask is offset. This offsetting causes the global blending mask to change, which in turn alters how the details are mixed. As a result, the object's appearance shifts as the blending of the details adjusts to the new position. ### Shader Setup #### General Shader Parameters When setting up the shader, the initial view displays the basic parameters, which can be further expanded for customization.
**Two key points to note:** 1. The shader is part of the **Dagor Material 2** set.
2. The **Add** button allows access to additional parameters.
#### Texture Assignment Textures are assigned to slots in the following order: - Slot 1 – Overlay - Slot 2 – Blending Mask - Slot 3 – Normal Map - Slot 4 – Detail1 Diffuse - Slot 5 – Detail1 Normal Map - Slot 6 – Detail2 Diffuse - Slot 7 – Detail2 Normal Map #### Shader Parameters The shader offers a range of parameters for customization: - `atlas_first_tile`/`atlas_last_tile`/`atlas_tile_u`/`atlas_tile_v`: Group of parameters that manages the offset of uv-mapping on the universal overlay. - `detail_tile_u/v`: Controls the tiling of details. - `detail_scroll_u/v`: Adjusts the scrolling of details. - `mask_gamma_start/end`: Configures the transparency of Detail1 through Detail2. - `mask_tile_u/v`: Controls the tiling of the blending mask. - `mask_scroll_u/v`: Adjusts the scrolling of the blending mask. - `draw_grass`: Enables grass growth on the object. - `detail2_colored`: Allows to colorize Detail2 in the *Dagor Editor* tool (an outdated feature rarely used). - `top_projection`: Group of parameters that allows to project Detail2 above the Detail1 using global and local normals. - `top_projection_from` - `top_projection_to` - `top_projection_detail2_modulate` - `use_painting`: Group of parameters used for colorizing of details. - `painting_line` - `paint1stdetail` - `detail2_combined_normal`: Blends details' normal maps. - `micro_detail_layer1`: Group of parameters for microdetails. - `micro_detail_layer2` - `micro_detail_layer1_uv_scale` - `micro_detail_layer2_uv_scale` - `material_shadow_influence`: Controls shadow strength from the details. - `material_shadow_scale`: Adjusts the shadow size from the details. #### Offset of UV-mapping on the Universal Overlay To use the universal overlay effectively, remember the rule stated earlier: ```{important} All exterior surfaces of the building, except the roof, should be mapped to **cell 8**. The shader will automatically shift the mapping across **cells 9-15**. ``` The shader and universal overlay/mask are designed to automate content variation. By mapping the exterior of the building to **cell 8**, the shader will automatically shift the mapping to other cells, creating varied appearances without manual intervention. For example, if the walls and exterior parts of the house are mapped to **cell 8**:
The shader will automatically shift the mapping to **cell 9**: ```{eval-rst} .. video:: _images/rendinst_layered_43.webm :width: 80% ``` However, this automatic shifting requires specific parameters to be set. The universal overlay cells are as follows: - **cells 0-1**: For internal materials (walls with wallpaper/stucco). - **cell 2**: For decorative wooden roof elements. - **cells 3, 6, 7**: Currently unused. - **cells 4-5**: For roof decoration: **cell 4** for sloped roofs, **cell 5** for flat roofs. - **cells 8-15**: For decorating external parts of buildings.
To enable the shader to recognize and use these cells, the following parameters must be set: - `atlas_first_tile`: Start cell number for the mapping jump (usually **cell 8**). - `atlas_last_tile`: End cell number for the mapping jump (usually **cell 15**). - `atlas_tile_u`: Horizontal jump size in fractions of the texture width (e.g., 0.5). - `atlas_tile_v`: Vertical jump size in fractions of the texture height (e.g., 0.125). For the universal overlay to work correctly, you need to add these parameters using the **Add** button and set their values as follows: - `atlas_first_tile = 8` - `atlas_last_tile = 15` - `atlas_tile_u = 0.5` - `atlas_tile_v = 0.125` If these parameters are not set, the overlay will not shift correctly: ```{eval-rst} .. video:: _images/rendinst_layered_45.webm :width: 80% ``` In some cases, you may want to limit the use of certain cells. For example, wooden walls may not look good with stucco cracks and stains.
In such cases, set `atlas_last_tile = 10` to limit the jump to **cells 8-10**, which are more neutral: ```{eval-rst} .. video:: _images/rendinst_layered_47.webm :width: 80% ``` #### Detail Tiling Settings The tiling of details is controlled by the parameters: - `detail1/2_tile_u`: Adjusts the horizontal tiling of Detail1/2. - `detail1/2_tile_v`: Adjusts the vertical tiling of Detail1/2. By default, if these parameters are not set, they are assumed to be `1`. These settings are useful when: - The texels of the detail textures do not match, or - You want to adjust the display size of one detail relative to another. Here's an example of a house corner with the brick detail tiling set to `1` and the stucco detail tiling set to `0.5`:
The stucco appears blurry because its texel size is twice as large as the brick. Comparing the textures:

Modern Brick Brown Dirt (1024x1024)

Stucco New (512x512)

Increasing the stucco tile by 2x for both axes corrects this:
However, you might want a more refined stucco finish without large, clear holes. Setting the stucco tile to `2.54` achieves this:
The result is a house with a more refined stucco texture. However, be cautious, as extreme mismatches in texel sizes can lead to visual inconsistencies, particularly on less detailed surfaces like roofs. For instance, moss with a tile of `5.35` on a roof with a tile of 1 shows a noticeable mismatch:
To avoid such issues, always check texture sizes and the final result, ensuring there's no visible mismatch. ```{important} It's better to use non-integer tiling parameters (e.g., `1.54` instead of `1.5`, `2.35` instead of `2`). This reduces the visibility of the tiling pattern. ``` #### Disabling Tiling (detail1/2_scroll_u=-1) ```{important} Use just `detail1/2_scroll_u` to disable detail scaling. Don't use `scroll_v`, because it's incorrect and leads to visual bugs. ``` When scaling an object, tiling is often automatically applied based on the object's size, which can cause unwanted effects on unique UV mappings. For instance, scaling a stone object 100 times can make the details tile 100 times more frequent. **Example:**

100% scale

1000% scale

The technology is useful, because it saves us texel when scaling objects. However, a problem arises when we use such a shader on objects with unique UV. For example, we take a bunch of tires and make them “dusty”. Detail1 mapping is scaled when we scale the model of tires. But uv-mapping for this model is unique - we get visual bug. **Example:**

100% scale:

200% scale:

To prevent this, set `detail1_scroll_u=-1` to disable tiling for a specific detail when scaling is not desired.

200% scale with detail1_scroll_u=-1


### Configuring Detail Offset Control the horizontal and vertical offsets of details using: - `detail1/2_scroll_u` – Horizontal offset. - `detail1/2_scroll_v` – Vertical offset. Default values are `1`. Use these settings when precise alignment of details is necessary, such as aligning bricks in a wall section.


To do this, we set the brick parameters `detail1_scroll_u` and `detail1_scroll_v` to 0. What happens if we don't set these parameters? For example, `detail1_scroll_u`:
We ended up with horizontal shifts in the texture, resulting in uneven seams: in some places, the bricks vary in thickness, while in others, they appear unusually short. It's easy to imagine what happens if `detail1_scroll_v=0` is not set – complete misalignment of the fence edges."
```{important} Don't set these parameters without a good reason. They limit the number of variations you can achieve with your objects, so only use them when absolutely necessary. ``` ### Configuring of Details Blending Details are blended with the final blending mask, that represents a result of combining of several masks (details' heightmaps, exclusion mask, blending mask). We can modify this final blending mask by raising it to the needed power - we call it Gamma. There're 2 parameters: - `mask_gamma_start` – smallest needed gamma value. - `mask_gamma_end` – largest needed gamma value. We specify them both and `Dagor` chooses final gamma value between them randomly. Our final blending mask is raised to the power of this gamma value. Imagine final blending mask as a set of values – not colors. Black is `0`. White is `1`. Nearly white is, for instance, `0.95`. Raise `0.95` to the power of `2` and you'll get `0.9`. But raise `0.95` to the power of `0.2` and you'll get `0.9898`. It means that the smaller gamma values you specify – the brighter final blending mask you get. And more of Detail2 will be seen. On the contrary the larger gamma values you specify – the darker final blending mask you get. And more of detail1 will be seen. For instance, we want to reduce the visibility of the stucco on this house.
We assign the blend mask `gypsum_m` as it is the most "gentle" and gray, and we get the following result:
This is not sufficient. Let's set the following parameter values: - `mask_gamma_start = 4` - `mask_gamma_end = 8`
### Configuring Blend Mask Tiling This is controlled by the following parameters: - `mask_tile_u` – horizontal tiling of the blend mask; - `mask_tile_v` – vertical tiling of the blend mask. If these parameters are not set, they default to `1`. These parameters are usually needed to add variety to the appearance of an object. For example, here is our fence with the `gypsum_m` mask tiling set to `1`.
We want to reduce the size of the moss patches but increase their frequency. Let's set the mask tiling to `8.75`.
Unlike detail textures, the blend mask can be tiled non-proportionally. For instance, we can keep the vertical tiling at `1` while setting the horizontal tiling to `8.75`.
There will be no stretching of textures – we've simply created narrower and taller blend areas, as the mask tiles nearly 9 times more frequently horizontally than vertically. This technique can be useful when you need to create "leaks" using the Detail2 layer. ### Configuring Blend Mask Offset This is controlled by the following parameters: - `mask_scroll_u` – horizontal offset of the mask; - `mask_scroll_v` – vertical offset of the mask. By default (when not added to the material), these values are set to `1`. In the shader textures documentation, we mentioned that the blend mask offsets each time the object's position in space changes. Let's revisit how this looks with the overlay removed: ```{eval-rst} .. video:: _images/rendinst_layered_73.webm :width: 80% ``` These parameters increase the offset value, allowing for more varied blending of detail layers. Let's set these values to `0.5`. ```{eval-rst} .. video:: _images/rendinst_layered_74.webm :width: 80% ``` As you can see, the changes between blends became - sometimes more visible, - sometimes less visible. This is because the offsets of our blend mask became smaller. And uv-mapping stays more time on areas that it used to jump over with offset=1. We rarely use this parameter, because it's much easier to random blending change with more varied blending mask. ```{important} If you need to disable blend mask jumps, use the parameter `mask_tile_u=-1`. Not the scroll, but the tiling, specifically `mask_tile_u` set to `-1`. ``` ### Growing Grass on an Object If you need to "grow" grass on a rendinst geometry, set the `draw_grass` parameter in its material to a value of `1`. It's important to apply this on a nearly horizontal surface to prevent grass from growing on walls. ### Coloring the Detail2 in the daEditor (Deprecated) In the initial version of the coloring process, the color range was specified directly in the [*daEditor*](../../dagor-tools/daeditor/daeditor/daeditor.md). Only the Detail2 (the "upper" one) was colored, using a simple multiply operation. This required careful selection of colors to avoid over-saturation. The process was as follows: 1. The parameter `detail2_colored` was set to `1`. 2. In the *daEditor*, a color gradient was applied set in the **Properties > Textures > Rendinst 2-nd detail color** section. ### Top Projection - Projecting the Detail2 using Local and Global Normals There are scenarios where you need to add snow, sand, or moss to an object. For this, the `top_projection` feature is used. It projects the Detail2 layer on top of the Detail1 at a specified angle. For instance, on a stone surface, you can project snow on top of the stone detail. The parameters for this feature are: - `top_projection` – 0/1 (off/on); - `top_projection_from` – the angle at which the projected detail starts to appear (at a minimal amount); - `top_projection_to` – the angle at which the projected detail fully covers the object; - `top_projection_detail2_modulate` – the intensity of the projection's effect on the Detail2 layer. The function takes into account all normals: 1. Global geometry normals. 2. Global normal map normals. 3. Local detail layer normals. The function also takes into consideration the masks: 1. Exclusion mask (overlay alpha) – if set to black, the Detail2 layer will not appear. 2. Blend mask. 3. However, it does NOT take into account the heightmap of the Detail1 layer, as it could be very bright (e.g., stucco). If it were considered, projection would be difficult. This is a procedural function – rotate the object as you like, and the Detail2 layer will cover it according to the specified rules, from the appropriate side. ```{eval-rst} .. video:: _images/rendinst_layered_75.webm :width: 80% ``` Let's examine how these parameters work. ### top_projection This parameter has values `0` and `1` (off/on). #### top_projection_from Specifies the angle at which the projected detail begins to appear (at a minimal amount). The angle is specified not in degrees but in the sine of the angle. For example, if you want the snow to start at 45 degrees, take sin(45) = 0.7 and enter `0.7` as the value.

Parameter value = 0.7

Parameter value = 0.1

#### top_projection_to Specifies the angle at which the projected detail fully covers the object. Again, this is specified in the sine of the angle. The difference is less noticeable than with the starting parameter – it's more apparent on flatter objects than on steep ones.

Parameter value = 0.5

Parameter value = 1

#### top_projection_detail2_modulate Specifies the intensity of the overlay's effect on the second detail layer. If you want pure white snow, set the value to `0`. If you want it to appear dirtier, increase the parameter up to `1`.

Parameter value = 0.2

Parameter value = 1

In this example, the difference is not very noticeable because the overlay itself is grayish. ### Blending with Landscape Heightmap To improve the visual integration of objects with the landscape, you can use special parameters that allow detail layers of terrain and assets to blend based on the intersection mask of the asset with the heightmap. This is particularly useful for piles, rocks, cliffs, and similar objects. ```{important} This is a computationally expensive technique, so use it sparingly. ``` **Examples:**

Without blend

With blend

Blending parameters: - `enable_hmap_blend=1` (default: `0`) – Enables/disables heightmap blending. - `hmap_blend_height=0.1` (default: `0.1`) – Height of the blending. - `ri_hmap_min_blend_angle=40` – Angle where landscape blending starts to fade out. (90 degrees is upward). - `ri_hmap_max_blend_angle=50` – Angle where landscape blending completely fades out. (90 degrees is upward). - `ri_hmap_blend_angle_vertex_normal_amount=0.75` – Floating value between texel (0) or vertex (1) normals as dominant for the blend mask. The higher the vertex normal dominance, the lower the chance of the projection appearing on triangles with a vertical normal, even if the texel normal is directed upward. - `ri_hmap_blend_check_holes=1` – Cuts the blend band at the intersection with heightmap holes created using `ground_hole:gameobj`. This parameter should be added **only** for rocks (or other objects that intersect between the heightmap and underground space), as unnecessary checks can significantly impact performance. GPU objects and grass now correctly generate on objects with Heightmap Blending. #### Global Shader Variables These are shader parameters initialized in `gamedata/.../scene.blk2` within the `shader_vars__vars:object` block: - `ri_hmap_blend_sharpness=8.0` (default: `8.0`) – The higher the value, the sharper the blend boundary. - `ri_hmap_perlin_tex_scale=0.25` (default: `0.25`) – Perlin noise tiling multiplier. - `ri_hmap_min_height=0.5` (default: `0.5`) – Minimum blend height, overriding the shader parameter if necessary. **Examples:**

ri_hmap_blend_sharpness=2

ri_hmap_blend_sharpness=4

ri_hmap_blend_sharpness=8

ri_hmap_perlin_tex_scale=0.01

ri_hmap_perlin_tex_scale=0.25

ri_hmap_perlin_tex_scale=4

ri_hmap_min_height=0.1

ri_hmap_min_height=0.5

ri_hmap_min_height=3

#### Static Shader Variables These are additional shader parameters that adjust the influence of global shader variables for a specific asset: - `ri_hmap_blend_sharpness_mul=1.0` – Multiplier for `ri_hmap_blend_sharpness`. - `ri_hmap_perlin_tex_scale_mul=1.0` – Multiplier for `ri_hmap_perlin_tex_scale`. - `ri_hmap_min_height_mul=1.0` – Multiplier for `ri_hmap_min_height`. ```{note} **Global and Static Shader Variables for ri_hmap** 1. Global shader variables affect all assets on the level with initialized blending. 2. If global shader variables are not initialized in `scene.blk`, the default values listed above are applied. 3. Static shader variables are multipliers of the global shader variables, initialized in `scene.blk`, for a specific asset. For example, if the global shader variable is set to: `ri_hmap_blend_sharpness=8.0` and the static shader variable for the asset `example.dag` is: `ri_hmap_blend_sharpness_mul=0.5` then the final `blend_sharpness` value for the asset `example.dag` on the level will be calculated as: `blend_sharpness = ri_hmap_blend_sharpness * ri_hmap_blend_sharpness_mul = 8.0 * 0.5 = 4` ```{important} The heightmap blending shader also works in [*daEditor*](../../dagor-tools/daeditor/daeditor/daeditor.md), but since global shader variables for the editor are initialized separately and not pulled from `scene.blk`, keep in mind that if you choose global shader variable values in `scene.blk` that differ from the defaults, the editor will display something different from what you'll see in the game. Instructions on how to configure global shader variables for the editor will be added to the article later. ``` ``` ### Procedural Detail Layer Coloring You can procedurally color any detail layer (only one at a time) using a palette. The palette is called `paint_colors.dds` and is located at `...\develop\assets\textures\colorize_textures`.
It consists of a set of single-pixel rows, each containing between 1 and 64 colors. In the coloring parameters, you specify a particular row for a specific object (from 0 to the last row in the texture). As the object moves across the map, each step will randomly shift the pixel within the selected coloring row. If multiple colors are specified, the object will be procedurally recolored. ```{eval-rst} .. video:: _images/rendinst_layered_87.webm :width: 80% ``` The coloring is applied based on the diffuse alpha (either the heightmap of the detail layer or simply a coloring mask for a unique texture). **Shader parameters:** - `use_painting` – Controls whether to color detail layers from the palette defined in the scene's shader variables. - `1` – color, - `0` – don't color; - values between `0.(01)` and `0.(9)` – partial coloring (multiplier). Coloring is based on the diffuse alpha, multiplied by the `use_painting` value. Values between `1.(0)1` and `1.(9)` also affect the coloring strength from 0.0 to 1.0 but disable random pixel selection from the coloring row when the object changes height – useful for consistently coloring modular skyscrapers, for example. - `painting_line` – The coloring row (from 0 to the last row in the texture). - `paint1stdetail` – Default value is `1` (colors the first detail layer). If you need to color the Detail2 layer, set the value to `-1`. ### Normal Blending There are situations where you need to reveal the underlying texture of the Detail1 layer (such as metal texture) through the Detail2 layer (such as paint). This is achieved using the `detail2_combined_normal` parameter, which has two possible values: `0` (disabled) or `1` (enabled).

Parameter value = 1

Parameter value = 0

### Microdetails ```{seealso} For information on microdetails, see [Microdetails on Assets](../about-assets/microdetails/microdetails.md). ``` ### Material Shadows Material shadows are generated based on the heightmap of detail layers. The following parameters are available: - `material_shadow_influence=0.5, 1, 0, 0` – Controls the "intensity" of shadows. Each component affects the corresponding detail layer. For instance, `0.5` affects the Detail1 layer, and `1` affects the Detail2 layer. The third and fourth components are not used in the `rendinst_layered` shader since it only supports two detail layers. However, in the `rendinst_perlin_layered` shader, the first three components are used, as it supports three detail layers. - `material_shadow_scale=2` – Acts as a multiplier for the "size" of the shadows. This applies to the shadows of all detail layers, as calculating it separately for each layer would be computationally expensive. **Example:**
## Shader Configuration Errors This occurs when you forget to set all the necessary values. For example, you might add the parameter `detail2_tile_u=2.54` but forget to include `detail2_tile_v=2.54`.
At first glance, this might seem insignificant (and this is often where the inspection ends, unfortunately). However, upon closer examination:
Such subtle issues are common and often go unnoticed until you closely inspect the settings. Sometimes, for instance, values like `detail2_tile_u=2.3` and `detail2_tile_v=2.4` are set, which only becomes noticeable upon careful examination. #### Sloped Roof Let's recall what a sloped roof tile looks like and how it reflects details:
Remember that each overlay tile contains a specifically designed blending mask.
Here's how this mask works with a house roof:
What happens if we leave the tiling parameters on the roof material? First, the roof will never land on the correct roof cell because `atlas_first_tile = 8`. Second, you'll end up with completely arbitrary overlays and blending. For example:
As you can see, the logic of material blending on the roof is completely lost. The materials blend across the entire surface as if they were on walls, not a roof. Now, let's see how the overlay appears on the roof. ```{eval-rst} .. video:: _images/rendinst_layered_97.webm :width: 80% ``` The result is completely abstract decorations, which have nothing to do with the intended metal sheets of the roof. This is precisely why the roof cell was designed to be universal and neutral. #### Flat Roof Let's recall how the flat roof cell looks and what it represents:
Here's its exclusion mask:
In this case, there's no need to show examples of incorrect atlas usage because there simply aren't any other cells that can fulfill the same function. #### Wooden Roof Trim Recall that the wooden trim has its own designated - **cell 2**:
When mapped correctly, the wooden planks look like this (Detail2 shows charred wood):
What happens when you apply overlay tiling parameters here? The mapping jumps to **cell 8**, and instead of smooth blending and a bronzed appearance, you get these bright planks smeared with tar: