UE 4.27 添加自定义 ShadingModel

2022-12-27
3 min read

基于 UE 4.27 的版本添加自定义的 ShadingModel ,大致分为两步:

1. 在 UE Editor 中添加自定义的 ShadingModel 入口。这样在创建材质时可以选择对应的 ShadingModel 。

2. 在 Shader 中对自定义的 ShadingModel 做渲染上的处理。比如自定义的 BxDF 函数,以及对光照的特殊处理等。

Editor 中添加 ShadingModel 入口

定义 ShadingModel 宏

首先是找到 EngineTypes.h 文件,在这里添加自定义 ShadingModel 对应的宏。

同时要注意下面红框中的内容,提示 ShadingModel 数量要小于等于 16 个,这样一来留给我们拓展的数量就有限了。

通过全局搜索,可以找到 ShadingModel 对应的宏都在哪里被引用了,比如 MSM_ThinTranslucent 的搜索结果:

照着这个引用方式进行处理就可以。

材质添加自定义数据

接下来为自定义的 ShadingModel 添加额外的自定义数据,会在创建材质时多出额外的连接节点。

找到 Material.cpp 文件 IsPropertyActive_Internal 的函数,添加如下代码:

在 MP_CustomData1 和 MP_CustomData0 中添加自定义 ShadingModel 的宏。

同时在 MaterialShared.cpp 文件的 GetAttributeOverrideForMaterial 函数中添加如下代码:

这一步是为自定义的数据添加相关的描述,添加好了之后编译引擎会得到如下效果。

ShadingModel 中有了我们定义的宏,连接节点中有了我们自定义的数据。

定义材质中的宏

添加了自定义 ShadingModel 的宏之外,还需要添加一个材质的宏,可以在材质中进行对该 ShadingModel 进行判断处理。

找到 HLSLMaterialTranslator.cpp 文件中的 GetMaterialEnvironment 函数,添加如下代码:

ShadingModel 渲染处理

添加 ShadingModelID

除了 Editor 中添加自定义 ShadingModel 的宏之外,还需要添加 ShadingModelID ,在后续的 Shader 中更多的是用这个宏来做判断,找到 ShadingCommmon.ush 文件,添加如下代码:

定义 ShadingModel 的调试显示颜色

在 ShadingCommon.ush 文件中还可以定义 ShadingModel 调试时的显示颜色,在 GetShadingModelColor 函数中,添加如下代码:

默认的 DEFAULT_LIT 调试颜色是绿色,上面定义的是黄色,效果如下:

自定义数据写入 GBuffer

要是 ShadingModel 中添加了自定义数据节点,要把它添加到 GBuffer 上,需要开启对应的选项才行,在 BassPassCommon.ush 文件中,添加如下代码:

WRITES_CUSTOMDATA_TO_GBUFFER 宏定义中添加预先定义好的数据。

除此之外,在 DeferredShadingModel.ush 文件的 HasCustomGBufferData 函数中增加自定义的 ShadingModelID 。

HasCustomGBufferData 函数是在解码 GBuffer 数据时根据 ShadingModelID 判断该 GBuffer 上有没有自定义的数据内容。

接下来就是往 GBuffer 中写入自定义数据,在 ShadingModelsMaterial.ush 文件的 SetGBufferForShadingModel 函数中添加如下代码:

该代码会先判断材质中有没有定义相应的宏,其次是判断 ShadingModelID 对不对,所以这两个宏都是必须要定义的。

通过 GetMaterialCustomData0 函数去获取蓝图连接点上的数据并且通过 saturate 函数将返回值限定在 0~1 之间,如果小于 0 就返回 0 ,如果大于 1 就返回 1 。

SetGBufferForShadingModel 函数会在 BasePassPixelShader.usf 中进行调用,写入 GBuffer 数据为延迟渲染使用。

自定义渲染效果 BxDF

接下来是最重要也最灵活多变的自定义渲染效果了,核心在于定义 BxDF 函数。

在 DeferredLightPixelShaders.usf 文件中的如下代码:

FRectTexture RectTexture = InitRectTexture(DeferredLightUniforms.SourceTexture);
float SurfaceShadow = 1.0f;

const float4 Radiance = GetDynamicLighting(WorldPosition, CameraVector, HairScreenSpaceData.GBuffer, HairScreenSpaceData.AmbientOcclusion, HairScreenSpaceData.GBuffer.ShadingModelID, LightData, LightAttenuation, Dither, PixelCoord, RectTexture, SurfaceShadow);

const float  Attenuation = ComputeLightProfileMultiplier(WorldPosition, DeferredLightUniforms.Position, -DeferredLightUniforms.Direction, DeferredLightUniforms.Tangent);

GeyDynamicLighting 函数中的 GetDynamicLightingSplit 函数会调用 IntegrateBxDF 来计算各个 ShadingModelID 对应的 BxDF 函数。

这里作为对比,定义了一个简单的 BxDF 函数,将漫反射颜色设置为 0 。

FDirectLighting CustomBxDF(FGBufferData GBuffer, half3 N, half3 V, half3 L, float Falloff, float NoL, FAreaLight AreaLight, FShadowTerms Shadow )
{
	BxDFContext Context;
	
	Init(Context, N, V, L);
	SphereMaxNoH(Context, AreaLight.SphereSinAlpha, true);
	
	Context.NoV = saturate(abs( Context.NoV ) + 1e-5);

	FDirectLighting Lighting;
	// Lighting.Diffuse  = AreaLight.FalloffColor * (Falloff * NoL) * Diffuse_Lambert( GBuffer.DiffuseColor );
	Lighting.Diffuse  = 0;
	
	Lighting.Specular = AreaLight.FalloffColor * (Falloff * NoL) * SpecularGGX(GBuffer.Roughness, GBuffer.SpecularColor, Context, NoL, AreaLight);
	Lighting.Transmission = 0;
	return Lighting;
}

在只有 DirectionalLight 光照情况下的效果如下:

可以看到高光效果还是有的,缺少了漫反射效果,整个物体都偏黑了。

作为对比,把高光颜色设置为 0 ,正常使用漫反射颜色效果如下:

可以看到有正常的颜色,但是没有了高光效果。

更多效果都可以在 BxDF 函数中去定义了,作为添加自定义 ShadingModel 的流程基本就完成了。

原创文章,转载请注明来源:    UE 4.27 添加自定义 ShadingModel


欢迎关注微信公众号:音视频开发进阶