Decals: Difference between revisions
| ThaiGrocer (talk | contribs) |  (These two count as broken by an update, as they are still included in the assets and the expected behavior is to have them fading out) | ||
| (60 intermediate revisions by 14 users not shown) | |||
| Line 1: | Line 1: | ||
| {{ | {{LanguageBar}} | ||
| | | {{tabs|Decals|goldsrc=1|source=1|main=source}} | ||
| }} | |||
| [[File:Decal06.jpg|thumb|A decal placed by a level designer.]] [[File:Decal03.jpg|thumb|Bullet impacts generated by the game.]] | [[File:Decal06.jpg|thumb|A decal placed by a level designer.]] [[File:Decal03.jpg|thumb|Bullet impacts generated by the game.]] | ||
| Line 19: | Line 19: | ||
| : Creates [[infodecal]] entities, which decal [[brush]]es and [[displacement]]s. | : Creates [[infodecal]] entities, which decal [[brush]]es and [[displacement]]s. | ||
| ; '''[[info_projecteddecal]]''' | ; '''[[info_projecteddecal]]''' | ||
| : An entity that  | : An entity that applies a dynamic decal similar to bullets or blood. The angle of projection can be set, allowing for distortion when on [[models]]. | ||
| ; '''{{hammer overlay}}''' | ; '''{{hammer overlay}}''' | ||
| :  | : [[info_overlay|Overlays]] are manipulatable decals on brushes and displacements that offer level designers more control. | ||
| ; '''[[info_overlay_transition]]''' | |||
| : A special type of overlay which is applied on the boundary between a [[water]] brush and a displacement. | |||
| [[infodecal]] and [[info_projecteddecal]] create [[edicts]] that last until the decal is sprayed (for unnamed decals, this is a split second after the map spawns). [[info_overlay]] only creates edicts if named. [[info_overlay_transition]] will never create edicts. | |||
| {{tip|If a decal entity has a name it will appear when [[use]]d (i.e. blank input) or on receiving the <code>activate</code> input, instead of when the map starts. The entity then disappears.}} | {{tip|If a decal entity has a name it will appear when [[use]]d (i.e. blank input) or on receiving the <code>activate</code> input, instead of when the map starts. The entity then disappears.}} | ||
| {{note|1=<span id="brush_decals"></span>Decals can also be made out of brushes or displacements using {{cmd|$decal}} materials, by having the decal be coplanar to the face that it is decaling. Due to [[VBSP]] performing a CGG pass on map geometry, however, decal brushes which aren't part of a separate [[bmodel]] (i.e. using [[func_illusionary]]) may not reliably appear in-game. Additionally, brush-based decals will use additional [[brushside]]s, and brush or displacement-based decals will have their own lightmap (allowing additional effects, such as [[normal mapping]]. | |||
| {{Confirm|[[func_detail]]-based decals will reliably show up on structural world brushes and brush entities, but what about on other func_details?}} | |||
| Decals can also be made out of [[MDL (Source)|MDLs]] this way, but may not blend as cleanly due to being vertex-lit.}} | |||
| === C++ === | === C++ === | ||
| Line 33: | Line 39: | ||
| Created in response to impacts, blood, and so on. Limited in number by the value of [[r_decals]], [[CRecipientFilter#Reliable vs. Unreliable|unreliable]] when transmitted, and not transmitted to players who connect after creation. | Created in response to impacts, blood, and so on. Limited in number by the value of [[r_decals]], [[CRecipientFilter#Reliable vs. Unreliable|unreliable]] when transmitted, and not transmitted to players who connect after creation. | ||
| {{note|Only materials registered in < | {{note|Only materials registered in <tt>scripts/decals_subrect.txt</tt> can be used!}} | ||
| ; < | ; <tt>CBaseEntity::DecalTrace([[trace_t]]* pTrace, const char* decalName)</tt> | ||
| : Shared code. The trace defines where the decal is projected from and its location. {{todo|How to avoid transmission being suppressed due to prediction?}} | : Shared code. The trace defines where the decal is projected from and its location. {{todo|How to avoid transmission being suppressed due to prediction?}} | ||
| ; < | ; <tt>C_BaseEntity::AddDecal( <lots of args> );</tt> | ||
| : Client function where the decal is actually created. In most cases you are better off just calling < | : Client function where the decal is actually created. In most cases you are better off just calling <tt>DecalTrace()</tt>. | ||
| ==== Static ==== | ==== Static ==== | ||
| Line 44: | Line 50: | ||
| Normally created by the level designer. These decals last forever and are always transmitted, including to players who connect later on. Any material can be used. | Normally created by the level designer. These decals last forever and are always transmitted, including to players who connect later on. Any material can be used. | ||
| ; < | ; <tt>engine->StaticDecal( Vector localOrigin, int decalIndex, int entityIndex, int modelIndex, bool lowpriority )</tt> | ||
| : Server only. This system does not appear use entities. | : Server only. This system does not appear use entities. | ||
| :; < | :; <tt>Vector localOrigin</tt> | ||
| :: Where the decal will be created, relative to the target's origin. | :: Where the decal will be created, relative to the target's origin. | ||
| :; < | :; <tt>int decalIndex</tt> | ||
| :: A value received from < | :: A value received from <tt>UTIL_PrecacheDecal()</tt>. ''Don't'' use <tt>GetDecalIndexForName()</tt>. | ||
| :; < | :; <tt>int entityIndex</tt> | ||
| :; < | :; <tt>int modelIndex</tt> | ||
| :: Information about the target entity. | :: Information about the target entity. | ||
| :; < | :; <tt>bool lowpriority</tt> | ||
| :: Low priority decals are not saved/restored, and can be "re-used on the client preferentially". | :: Low priority decals are not saved/restored, and can be "re-used on the client preferentially". | ||
| ; < | ; <tt>UTIL_PrecacheDecal(const char *filename, bool preload)</tt> | ||
| : Returns a decal index for use with < | : Returns a decal index for use with <tt>StaticDecal()</tt>. If this is in <tt>[[precache()]]</tt>, remember that it may be called statically; store the result in a global var. | ||
| == Creating == | == Creating == | ||
| Line 62: | Line 68: | ||
| : ''This section assumes that you understand the basics of [[Material Creation|material creation]].'' | : ''This section assumes that you understand the basics of [[Material Creation|material creation]].'' | ||
| Any material can be used as a decal. They generally have an [[alpha channel]] however, and can use some special parameters and/or the <code>[[#DecalModulate|DecalModulate]]</code> shader. | Any material can be used as a decal. They generally have an [[alpha channel]] however (using [[$translucent]] or [[$alphatest]]), and can use some special parameters and/or the <code>[[#DecalModulate|DecalModulate]]</code> shader. | ||
| < | <gallery mode=nolines widths=128px style="text-align:center;"> | ||
| decal04.jpg|Source RGB color channels | |||
| decal05.jpg|Source alpha channel | |||
| Decal06.jpg|Decal in the world | |||
| </gallery> | |||
| === Material parameters === | ===Material parameters=== | ||
| ; <tt>[[$decal]] <[[bool]]></tt> | |||
| ; < | : Prevents decal texture clipping. | ||
| :  | : {{note|This parameter is not exclusive to decal entities; any material can have $decal to modify how it is sorted when coplanar with another surface.}} | ||
| ; < | :{{confirm|Unnecessary for overlays outside of Hammer?}} | ||
| ; <tt>$decalscale <[[float]]></tt> | |||
| : Same as a brush face's texture scale value: the number of [[unit]]s that each [[texel]] covers. Normally 0.25 or lower. {{note|This defines the size of the decal's geometry, and cannot be changed after its creation. Use [[$basetexturetransform]] if you want motion.}} | : Same as a brush face's texture scale value: the number of [[unit]]s that each [[texel]] covers. Normally 0.25 or lower. {{note|This defines the size of the decal's geometry, and cannot be changed after its creation. Use [[$basetexturetransform]] if you want motion.}} | ||
| ; < | ; <tt>$modelmaterial <material></tt> | ||
| : A separate < | : A separate <tt>[[VertexLitGeneric]]</tt> material to that will replace this one if the decal hits a [[model]]. | ||
| ; < | ; <tt>$decalfadeduration <float></tt> | ||
| : Amount of time to spend fading out. Requires < | : Amount of time to spend fading out. Requires <tt>[[$vertexcolor]]</tt>. | ||
| ; < | {{bug|hidetested=1| This is broken in {{src13|4}}, decals will not fade out}} | ||
| ; <tt>$decalfadetime <float></tt> | |||
| {{bug|hidetested=1| This is broken in {{src13|4}}, decals will not fade out}} | |||
| : Delay before fadeout begins. | : Delay before fadeout begins. | ||
| ; < | ; <tt>$decalsecondpass <[[bool]]></tt> {{src13|not}} | ||
| : {{ | : ALWAYS render this decal on top of decals ''without'' this parameter. If two decals with this parameter intersect, they will treat each other how two without the parameter would. | ||
| ; < | :{{tip|This should be used for dynamically created decals only, to prevent them from [[z-fighting]] with mapper-placed decals. Mappers should use {{ent|info_overlay}}'s {{code|RenderOrder}} KV instead.}} | ||
| : {{ | ; <tt>$decaldynamicscale <bool></tt> {{src13|not}} | ||
| : Scales decal relative to camera distance | |||
| ; <tt>$fogscale <float></tt> | |||
| : {{L4D2|since}} Scales the amount of fog affecting the decal. Useful for making important decals stand out in foggy levels. | |||
| ; <tt>$splatter <bool></tt> | |||
| : {{CSGO|since}} {{todo|What does this do?}} | |||
| Standard decals and overlays do not support [[normal map]]s in all engine branches prior to CS:GO. The underlying brushes/displacements ''must'' have bumped lightmaps for the decal's normal map to have any effect. | |||
| : {{confirm|Is CS:GO the first game to support this feature?}} | |||
| : {{Workaround|Use a coplanar brush or displacement with {{cmd|$decal}} instead. See note above.}} | |||
| === DecalModulate === | === DecalModulate === | ||
| {{split|[[DecalModulate]]}} | |||
| For decals intended to mimic the look of pock marks or dents in a surface, the '''<code>DecalModulate</code>''' (a.k.a. mod2x) shader is especially suitable: it lightens destination pixels for every source pixel that is over mid-range gray (128) and darkens any destination pixels for every source pixel that is below mid-range gray. This effect can be used to give the impression of depth when applied to a surface. | For decals intended to mimic the look of pock marks or dents in a surface, the '''<code>DecalModulate</code>''' (a.k.a. mod2x) shader is especially suitable: it lightens destination pixels for every source pixel that is over mid-range gray (128) and darkens any destination pixels for every source pixel that is below mid-range gray. This effect can be used to give the impression of depth when applied to a surface. | ||
| {{bug|tested={{src13}}| In {{src13|4}}, when applied to models that aren't static props, shining any {{ent|env_projectedtexture}} on top of DecalModulate decal causes the latter to fade. A common example is pointing player's flashlight at bullet holes of prop entities. A workaround is using [[Modulate]] shader paired with the [[$mod2x]] VMT parameter enabled.}} | |||
| To begin, create a source image whose color channel will be used for the modulation's source values. Again, light values will lighten pixels they're drawn over, while dark values will darken the destination pixels. Mid-gray values will be treated as translucent. | To begin, create a source image whose color channel will be used for the modulation's source values. Again, light values will lighten pixels they're drawn over, while dark values will darken the destination pixels. Mid-gray values will be treated as translucent. | ||
| Next, create an alpha channel that defines a mask for the decal. Because modulation cannot have an exact middle value  | Next, create an alpha channel that defines a mask for the decal. Because modulation cannot have an exact middle value with DXTn compression, the mask is necessary to prevent "bordering" from occurring around the decal. You can avoid "bordering" without creating an alpha channel simply by using BGR888 format instead of DXT1/DXT5 when you create your VTF, at the cost of being triple the file size. | ||
| {{modernConfirm|Does I8 (greyscale) work, like with {{cmd|$detailblendmode|0}}, or does the shader require RGB data? I8 is the same size as DXT5, and but accurately represent all 256 possible modulation values (albeit without color).}} | |||
| {{note|It has also been observed in {{l4d2|4}} that DXT1 compressed textures using DecalModulate shader (such as blood splatters and "graffiti wall writing") with background colors RGB 124 126 124 are considered unmodulated, alleviating "bordering". L4D2's [[Vtex (Source 1)|vtex]] will output that specific color from the source image pixels are gray RGB 127 127 127. This solution may also apply to other engine branches (and other versions of vtex) that use RGB 124 126 124 as the color indicating lack of modulation in-game, but needs further testing.}} | |||
| {{todo|Try converting source images with background colors RGB 127 127 127 with vtex found in other games other than L4D2 and see if it results in a background with RGB 124 126 124. Also, does RGB 124 126 124 apply to other versions of the Source engine as unmodulated for DXT1 compression? It is possible that the upgraded shader has a range of a few values before modulation occurs.}} | |||
| <gallery mode=nolines widths=128px style="text-align:center;"> | |||
| decal01.jpg|Source RGB color channels | |||
| decal02.jpg|Source alpha channel | |||
| Decal03.jpg|Decal in the world | |||
| </gallery> | |||
| < | Finally, you must create a material that uses the <code>DecalModulate</code> shader. Neither <code>$translucent</code> nor <code>$decal</code> are needed this time. | ||
| {{tip|{{mono|DecalModulate}} is equivalent to the "Overlay" layer mode in image editors such as {{gimp|4}} or {{photoshop|4}}.}} | |||
| == See also == | == See also == | ||
| Line 105: | Line 133: | ||
| * [[info_projecteddecal]] | * [[info_projecteddecal]] | ||
| * [[info_overlay]] | * [[info_overlay]] | ||
| * [[$nodecal]] | |||
| [[Category:Material System]] | [[Category:Material System]] | ||
| [[Category:Glossary]] | [[Category:Glossary]] | ||
Latest revision as of 16:36, 3 September 2025
Decals are materials projected onto existing surfaces. They can be placed by the level designer, and are also generated by the engine for bullet impacts, blood, and other effects.
 Note:Each decal affects one entity only. (The world is one big entity.)
Note:Each decal affects one entity only. (The world is one big entity.)Decals are "sprayed" from a location and mark every surface in their path. For instance, a decal applied downward onto a staircase would cascade down onto the top (but not front) of each step.
Using
 Note:VertexLitGeneric decals won't show up on brushes, and LightmappedGeneric ones won't appear on models. Decals using
Note:VertexLitGeneric decals won't show up on brushes, and LightmappedGeneric ones won't appear on models. Decals using DecalModulate or $modelmaterial work everywhere (see below).Hammer
- Decal tool  
- Creates infodecal entities, which decal brushes and displacements.
- info_projecteddecal
- An entity that applies a dynamic decal similar to bullets or blood. The angle of projection can be set, allowing for distortion when on models.
- Overlay tool  
- Overlays are manipulatable decals on brushes and displacements that offer level designers more control.
- info_overlay_transition
- A special type of overlay which is applied on the boundary between a water brush and a displacement.
infodecal and info_projecteddecal create edicts that last until the decal is sprayed (for unnamed decals, this is a split second after the map spawns). info_overlay only creates edicts if named. info_overlay_transition will never create edicts.
 Tip:If a decal entity has a name it will appear when used (i.e. blank input) or on receiving the
Tip:If a decal entity has a name it will appear when used (i.e. blank input) or on receiving the activate input, instead of when the map starts. The entity then disappears. Note:Decals can also be made out of brushes or displacements using
Note:Decals can also be made out of brushes or displacements using $decal materials, by having the decal be coplanar to the face that it is decaling. Due to VBSP performing a CGG pass on map geometry, however, decal brushes which aren't part of a separate bmodel (i.e. using func_illusionary) may not reliably appear in-game. Additionally, brush-based decals will use additional brushsides, and brush or displacement-based decals will have their own lightmap (allowing additional effects, such as normal mapping.
 Confirm:func_detail-based decals will reliably show up on structural world brushes and brush entities, but what about on other func_details?
 Confirm:func_detail-based decals will reliably show up on structural world brushes and brush entities, but what about on other func_details?C++
Standard
Created in response to impacts, blood, and so on. Limited in number by the value of r_decals, unreliable when transmitted, and not transmitted to players who connect after creation.
 Note:Only materials registered in scripts/decals_subrect.txt can be used!
Note:Only materials registered in scripts/decals_subrect.txt can be used!- CBaseEntity::DecalTrace(trace_t* pTrace, const char* decalName)
- Shared code. The trace defines where the decal is projected from and its location. Todo: How to avoid transmission being suppressed due to prediction?
- C_BaseEntity::AddDecal( <lots of args> );
- Client function where the decal is actually created. In most cases you are better off just calling DecalTrace().
Static
Normally created by the level designer. These decals last forever and are always transmitted, including to players who connect later on. Any material can be used.
- engine->StaticDecal( Vector localOrigin, int decalIndex, int entityIndex, int modelIndex, bool lowpriority )
- Server only. This system does not appear use entities.
- Vector localOrigin
- Where the decal will be created, relative to the target's origin.
- int decalIndex
- A value received from UTIL_PrecacheDecal(). Don't use GetDecalIndexForName().
- int entityIndex
- int modelIndex
- Information about the target entity.
- bool lowpriority
- Low priority decals are not saved/restored, and can be "re-used on the client preferentially".
 
- UTIL_PrecacheDecal(const char *filename, bool preload)
- Returns a decal index for use with StaticDecal(). If this is in precache(), remember that it may be called statically; store the result in a global var.
Creating
- This section assumes that you understand the basics of material creation.
Any material can be used as a decal. They generally have an alpha channel however (using $translucent or $alphatest), and can use some special parameters and/or the DecalModulate shader.
Material parameters
- $decal <bool>
- Prevents decal texture clipping.
 Note:This parameter is not exclusive to decal entities; any material can have $decal to modify how it is sorted when coplanar with another surface. Note:This parameter is not exclusive to decal entities; any material can have $decal to modify how it is sorted when coplanar with another surface.
 Confirm:Unnecessary for overlays outside of Hammer? Confirm:Unnecessary for overlays outside of Hammer?
- $decalscale <float>
- Same as a brush face's texture scale value: the number of units that each texel covers. Normally 0.25 or lower.  Note:This defines the size of the decal's geometry, and cannot be changed after its creation. Use $basetexturetransform if you want motion. Note:This defines the size of the decal's geometry, and cannot be changed after its creation. Use $basetexturetransform if you want motion.
- $modelmaterial <material>
- A separate VertexLitGeneric material to that will replace this one if the decal hits a model.
- $decalfadeduration <float>
- Amount of time to spend fading out. Requires $vertexcolor.
 Bug: This is broken in
Bug: This is broken in  Source 2013, decals will not fade out
 Source 2013, decals will not fade out- $decalfadetime <float>
 Bug: This is broken in
Bug: This is broken in  Source 2013, decals will not fade out
 Source 2013, decals will not fade out- Delay before fadeout begins.
- $decalsecondpass <bool> (not in  ) )
- ALWAYS render this decal on top of decals without this parameter. If two decals with this parameter intersect, they will treat each other how two without the parameter would.
 Tip:This should be used for dynamically created decals only, to prevent them from z-fighting with mapper-placed decals. Mappers should use info_overlay's Tip:This should be used for dynamically created decals only, to prevent them from z-fighting with mapper-placed decals. Mappers should use info_overlay's- RenderOrderKV instead.
- $decaldynamicscale <bool> (not in  ) )
- Scales decal relative to camera distance
- $fogscale <float>
- (in all games since  ) Scales the amount of fog affecting the decal. Useful for making important decals stand out in foggy levels. ) Scales the amount of fog affecting the decal. Useful for making important decals stand out in foggy levels.
- $splatter <bool>
- (in all games since  ) Todo: What does this do? ) Todo: What does this do?
Standard decals and overlays do not support normal maps in all engine branches prior to CS:GO. The underlying brushes/displacements must have bumped lightmaps for the decal's normal map to have any effect.
 Confirm:Is CS:GO the first game to support this feature? Confirm:Is CS:GO the first game to support this feature?
 Workaround:Use a coplanar brush or displacement with Workaround:Use a coplanar brush or displacement with- $decalinstead. See note above.
DecalModulate
For decals intended to mimic the look of pock marks or dents in a surface, the DecalModulate (a.k.a. mod2x) shader is especially suitable: it lightens destination pixels for every source pixel that is over mid-range gray (128) and darkens any destination pixels for every source pixel that is below mid-range gray. This effect can be used to give the impression of depth when applied to a surface.
 Bug: In
Bug: In  Source 2013, when applied to models that aren't static props, shining any env_projectedtexture on top of DecalModulate decal causes the latter to fade. A common example is pointing player's flashlight at bullet holes of prop entities. A workaround is using Modulate shader paired with the $mod2x VMT parameter enabled.  (tested in:
 Source 2013, when applied to models that aren't static props, shining any env_projectedtexture on top of DecalModulate decal causes the latter to fade. A common example is pointing player's flashlight at bullet holes of prop entities. A workaround is using Modulate shader paired with the $mod2x VMT parameter enabled.  (tested in:  )
)To begin, create a source image whose color channel will be used for the modulation's source values. Again, light values will lighten pixels they're drawn over, while dark values will darken the destination pixels. Mid-gray values will be treated as translucent.
Next, create an alpha channel that defines a mask for the decal. Because modulation cannot have an exact middle value with DXTn compression, the mask is necessary to prevent "bordering" from occurring around the decal. You can avoid "bordering" without creating an alpha channel simply by using BGR888 format instead of DXT1/DXT5 when you create your VTF, at the cost of being triple the file size.
 Confirm:Does I8 (greyscale) work, like with
 Confirm:Does I8 (greyscale) work, like with $detailblendmode 0, or does the shader require RGB data? I8 is the same size as DXT5, and but accurately represent all 256 possible modulation values (albeit without color). Note:It has also been observed in
Note:It has also been observed in  Left 4 Dead 2 that DXT1 compressed textures using DecalModulate shader (such as blood splatters and "graffiti wall writing") with background colors RGB 124 126 124 are considered unmodulated, alleviating "bordering". L4D2's vtex will output that specific color from the source image pixels are gray RGB 127 127 127. This solution may also apply to other engine branches (and other versions of vtex) that use RGB 124 126 124 as the color indicating lack of modulation in-game, but needs further testing.
 Left 4 Dead 2 that DXT1 compressed textures using DecalModulate shader (such as blood splatters and "graffiti wall writing") with background colors RGB 124 126 124 are considered unmodulated, alleviating "bordering". L4D2's vtex will output that specific color from the source image pixels are gray RGB 127 127 127. This solution may also apply to other engine branches (and other versions of vtex) that use RGB 124 126 124 as the color indicating lack of modulation in-game, but needs further testing.Finally, you must create a material that uses the DecalModulate shader. Neither $translucent nor $decal are needed this time.
 Tip:DecalModulate is equivalent to the "Overlay" layer mode in image editors such as
Tip:DecalModulate is equivalent to the "Overlay" layer mode in image editors such as  GIMP or
 GIMP or  Adobe Photoshop.
 Adobe Photoshop.






























 Split
 Split

