833 QSSGMaterialVertexPipeline &vertexShader,
834 const QSSGShaderDefaultMaterialKey &inKey,
835 const QSSGShaderDefaultMaterialKeyProperties &keyProps,
836 const QSSGShaderFeatures &featureSet,
837 const QSSGRenderGraphObject &inMaterial,
838 const QSSGUserShaderAugmentation &shaderAugmentation,
839 QSSGShaderLibraryManager &shaderLibraryManager)
841 QSSGShaderMaterialAdapter *materialAdapter = getMaterialAdapter(inMaterial);
842 auto hasCustomFunction = [&shaderLibraryManager, materialAdapter](
const QByteArray &funcName) {
843 return materialAdapter->hasCustomShaderFunction(QSSGShaderCache::ShaderType::Fragment,
845 shaderLibraryManager);
848 auto channelStr = [](
const QSSGShaderKeyTextureChannel &chProp,
const QSSGShaderDefaultMaterialKey &inKey) -> QByteArray {
850 switch (chProp.getTextureChannel(inKey)) {
851 case QSSGShaderKeyTextureChannel::R:
854 case QSSGShaderKeyTextureChannel::G:
857 case QSSGShaderKeyTextureChannel::B:
860 case QSSGShaderKeyTextureChannel::A:
867 auto maskVariableByVertexColorChannel = [&fragmentShader, keyProps, inKey](
const QByteArray &maskVariable,
const QSSGRenderDefaultMaterial::VertexColorMask &maskEnum ){
868 if (keyProps.m_vertexColorsMaskEnabled.getValue(inKey)) {
869 if ( keyProps.m_vertexColorRedMask.getValue(inKey) & maskEnum )
870 fragmentShader <<
" " << maskVariable <<
" *= qt_vertColorMask.r;\n";
871 else if ( keyProps.m_vertexColorGreenMask.getValue(inKey) & maskEnum )
872 fragmentShader <<
" " << maskVariable <<
" *= qt_vertColorMask.g;\n";
873 else if ( keyProps.m_vertexColorBlueMask.getValue(inKey) & maskEnum )
874 fragmentShader <<
" " << maskVariable <<
" *= qt_vertColorMask.b;\n";
875 else if ( keyProps.m_vertexColorAlphaMask.getValue(inKey) & maskEnum )
876 fragmentShader <<
" " << maskVariable <<
" *= qt_vertColorMask.a;\n";
880 generateFragmentDefines(fragmentShader, inKey, keyProps, materialAdapter, shaderLibraryManager, shaderAugmentation);
885 const PassRequirmentsState passRequirmentState(inKey, keyProps, featureSet, samplerState, shaderAugmentation);
887 const bool hasCustomVert = materialAdapter->hasCustomShaderSnippet(QSSGShaderCache::ShaderType::Vertex);
889 const int viewCount = featureSet.isSet(QSSGShaderFeatures::Feature::DisableMultiView)
890 ? 1 : keyProps.m_viewCount.getValue(inKey);
893 if (passRequirmentState.numMorphTargets > 0 || hasCustomVert) {
894 vertexShader.addDefinition(QByteArrayLiteral(
"QT_MORPH_MAX_COUNT"),
895 QByteArray::number(passRequirmentState.numMorphTargets));
897 if ((offset = keyProps.m_targetPositionOffset.getValue(inKey)) < UINT8_MAX) {
898 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_POSITION_OFFSET"),
899 QByteArray::number(offset));
901 if ((offset = keyProps.m_targetNormalOffset.getValue(inKey)) < UINT8_MAX) {
902 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_NORMAL_OFFSET"),
903 QByteArray::number(offset));
905 if ((offset = keyProps.m_targetTangentOffset.getValue(inKey)) < UINT8_MAX) {
906 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_TANGENT_OFFSET"),
907 QByteArray::number(offset));
909 if ((offset = keyProps.m_targetBinormalOffset.getValue(inKey)) < UINT8_MAX) {
910 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_BINORMAL_OFFSET"),
911 QByteArray::number(offset));
913 if ((offset = keyProps.m_targetTexCoord0Offset.getValue(inKey)) < UINT8_MAX) {
914 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_TEX0_OFFSET"),
915 QByteArray::number(offset));
917 if ((offset = keyProps.m_targetTexCoord1Offset.getValue(inKey)) < UINT8_MAX) {
918 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_TEX1_OFFSET"),
919 QByteArray::number(offset));
921 if ((offset = keyProps.m_targetColorOffset.getValue(inKey)) < UINT8_MAX) {
922 vertexShader.addDefinition(QByteArrayLiteral(
"QT_TARGET_COLOR_OFFSET"),
923 QByteArray::number(offset));
928 if (shaderAugmentation.hasUserAugmentation())
929 fragmentShader << shaderAugmentation.preamble;
932 for (
const auto &u : shaderAugmentation.propertyUniforms)
933 fragmentShader.addUniform(u.name, u.typeName);
936 const bool hasCustomFrag = materialAdapter->hasCustomShaderSnippet(QSSGShaderCache::ShaderType::Fragment);
937 const bool usesSharedVar = materialAdapter->usesSharedVariables();
939 vertexShader.beginFragmentGeneration(shaderLibraryManager, passRequirmentState.oitMethod);
942 vertexShader.generateDepth();
943 fragmentShader.addUniform(
"qt_shadowDepthAdjust",
"vec2");
948 vertexShader.generateShadowWorldPosition(inKey);
951 if (hasCustomFrag && materialAdapter->isUnshaded()) {
965 fragmentShader.addUniform(
"qt_material_emissive_color",
"vec3");
967 fragmentShader.addUniform(
"qt_material_base_color",
"vec4");
968 fragmentShader.addUniform(
"qt_material_properties",
"vec4");
969 fragmentShader.addUniform(
"qt_material_properties2",
"vec4");
970 fragmentShader.addUniform(
"qt_material_properties3",
"vec4");
972 fragmentShader.addUniform(
"qt_material_properties4",
"vec4");
973 if (!hasCustomFrag) {
975 fragmentShader.addUniform(
"qt_material_attenuation",
"vec4");
976 fragmentShader.addUniform(
"qt_material_thickness",
"float");
979 fragmentShader.addUniform(
"qt_material_properties5",
"vec4");
980 fragmentShader.addUniform(
"qt_material_clearcoat_normal_strength",
"float");
981 fragmentShader.addUniform(
"qt_material_clearcoat_fresnel_power",
"float");
985 vertexShader.generateVertexColor(inKey);
987 fragmentShader.append(
" vec4 qt_vertColorMask = vec4(1.0);");
988 fragmentShader.append(
" vec4 qt_vertColor = vec4(1.0);");
992 vertexShader.generateViewVector(inKey);
993 if (keyProps.m_usesProjectionMatrix.getValue(inKey)) {
995 fragmentShader.addUniformArray(
"qt_projectionMatrix",
"mat4", viewCount);
997 fragmentShader.addUniform(
"qt_projectionMatrix",
"mat4");
999 if (keyProps.m_usesInverseProjectionMatrix.getValue(inKey)) {
1001 fragmentShader.addUniformArray(
"qt_inverseProjectionMatrix",
"mat4", viewCount);
1003 fragmentShader.addUniform(
"qt_inverseProjectionMatrix",
"mat4");
1005 vertexShader.generateWorldNormal(inKey);
1006 vertexShader.generateWorldPosition(inKey);
1009 bool genTangent =
false;
1010 bool genBinormal =
false;
1011 vertexShader.generateVarTangentAndBinormal(inKey, genTangent, genBinormal);
1014 QSSGRenderableImage::Type id = QSSGRenderableImage::Type::Unknown;
1019 id = samplerState.hasImage(QSSGRenderableImage::Type::Bump) ? QSSGRenderableImage::Type::Bump : QSSGRenderableImage::Type::Normal;
1020 }
else if (samplerState.hasImage(QSSGRenderableImage::Type::ClearcoatNormal)) {
1022 id = QSSGRenderableImage::Type::ClearcoatNormal;
1026 id = QSSGRenderableImage::Type::Height;
1029 if (id > QSSGRenderableImage::Type::Unknown) {
1030 samplerState.generateImageUVAndSampler(id, vertexShader, fragmentShader, inKey,
true);
1031 fragmentShader <<
" vec2 dUVdx = dFdx(" << samplerState.fragCoordsName(id) <<
");\n"
1032 <<
" vec2 dUVdy = dFdy(" << samplerState.fragCoordsName(id) <<
");\n";
1033 fragmentShader <<
" qt_tangent = (dUVdy.y * dFdx(qt_varWorldPos) - dUVdx.y * dFdy(qt_varWorldPos)) / (dUVdx.x * dUVdy.y - dUVdx.y * dUVdy.x);\n"
1034 <<
" qt_tangent = qt_tangent - dot(qt_world_normal, qt_tangent) * qt_world_normal;\n"
1035 <<
" qt_tangent = normalize(qt_tangent);\n";
1039 fragmentShader <<
" qt_binormal = cross(qt_world_normal, qt_tangent);\n";
1043 fragmentShader.append(
"#if QSHADER_HLSL && QSHADER_VIEW_COUNT >= 2");
1044 fragmentShader.append(
" const float qt_facing = 1.0;");
1045 fragmentShader.append(
"#else");
1046 fragmentShader.append(
" const float qt_facing = gl_FrontFacing ? 1.0 : -1.0;");
1047 fragmentShader.append(
"#endif");
1048 fragmentShader.append(
" qt_world_normal *= qt_facing;\n");
1050 fragmentShader.append(
" qt_tangent *= qt_facing;");
1051 fragmentShader.append(
" qt_binormal *= qt_facing;");
1056 if (hasCustomFrag) {
1062 fragmentShader <<
" float qt_customOcclusionAmount = 1.0;\n";
1063 fragmentShader <<
" float qt_customIOR = 1.5;\n";
1064 fragmentShader <<
" float qt_customSpecularAmount = 0.5;\n";
1065 fragmentShader <<
" float qt_customSpecularRoughness = 0.0;\n";
1066 fragmentShader <<
" float qt_customMetalnessAmount = 0.0;\n";
1067 fragmentShader <<
" float qt_customFresnelPower = 5.0;\n";
1068 fragmentShader <<
" vec4 qt_customBaseColor = vec4(1.0);\n";
1069 fragmentShader <<
" vec3 qt_customEmissiveColor = vec3(0.0);\n";
1071 fragmentShader <<
" float qt_customClearcoatAmount = 0.0;\n";
1072 fragmentShader <<
" float qt_customClearcoatFresnelPower = 5.0;\n";
1073 fragmentShader <<
" float qt_customClearcoatRoughness = 0.0;\n";
1074 fragmentShader <<
" vec3 qt_customClearcoatNormal = qt_world_normal;\n";
1076 fragmentShader <<
" float qt_customClearcoatFresnelScale = 1.0;\n";
1077 fragmentShader <<
" float qt_customClearcoatFresnelBias = 0.0;\n";
1081 fragmentShader <<
" float qt_customFresnelScale = 1.0;\n";
1082 fragmentShader <<
" float qt_customFresnelBias = 0.0;\n";
1086 fragmentShader <<
" float qt_customTransmissionFactor = 0.0;\n";
1087 fragmentShader <<
" float qt_customThicknessFactor = 0.0;\n";
1088 fragmentShader <<
" vec3 qt_customAttenuationColor = vec3(1.0);\n";
1089 fragmentShader <<
" float qt_customAttenuationDistance = 0.0;\n";
1092 fragmentShader <<
" QT_SHARED_VARS qt_customShared;\n";
1095 vertexShader.generateUVCoords(0, inKey);
1096 vertexShader.generateUVCoords(1, inKey);
1098 fragmentShader <<
" qt_customMain(qt_customBaseColor,\n"
1099 <<
" qt_customEmissiveColor,\n"
1100 <<
" qt_customMetalnessAmount,\n"
1101 <<
" qt_customSpecularRoughness,\n"
1102 <<
" qt_customSpecularAmount,\n"
1103 <<
" qt_customFresnelPower,\n"
1104 <<
" qt_world_normal,\n"
1106 <<
" qt_binormal,\n"
1107 <<
" qt_texCoord0,\n"
1108 <<
" qt_texCoord1,\n"
1109 <<
" qt_view_vector,\n"
1110 <<
" qt_customIOR,\n"
1111 <<
" qt_customOcclusionAmount";
1113 fragmentShader <<
",\n qt_customClearcoatAmount,\n"
1114 <<
" qt_customClearcoatFresnelPower,\n"
1115 <<
" qt_customClearcoatRoughness,\n"
1116 <<
" qt_customClearcoatNormal";
1118 fragmentShader <<
",\n qt_customClearcoatFresnelScale,\n"
1119 <<
" qt_customClearcoatFresnelBias";
1123 fragmentShader <<
",\n qt_customFresnelScale,\n"
1124 <<
" qt_customFresnelBias";
1127 fragmentShader <<
",\n qt_customTransmissionFactor,\n"
1128 <<
" qt_customThicknessFactor,\n"
1129 <<
" qt_customAttenuationColor,\n"
1130 <<
" qt_customAttenuationDistance";
1133 fragmentShader <<
"\n, qt_customShared);\n";
1135 fragmentShader <<
");\n";
1137 fragmentShader <<
" vec4 qt_diffuseColor = qt_customBaseColor * qt_vertColor;\n";
1138 fragmentShader <<
" vec3 qt_global_emission = qt_customEmissiveColor;\n";
1139 fragmentShader <<
" float qt_iOR = qt_customIOR;\n";
1141 fragmentShader <<
" vec4 qt_diffuseColor = qt_material_base_color * qt_vertColor;\n";
1142 fragmentShader <<
" vec3 qt_global_emission = qt_material_emissive_color;\n";
1144 fragmentShader <<
" float qt_iOR = qt_material_specular.w;\n";
1147 const bool hasCustomIblProbe = hasCustomFrag && hasCustomFunction(QByteArrayLiteral(
"qt_iblProbeProcessor"));
1154 vertexShader.generateLightmapUVCoords(inKey);
1155 fragmentShader.addFunction(
"lightmap");
1164 fragmentShader.addInclude(
"parallaxMapping.glsllib");
1165 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Height, vertexShader, fragmentShader, inKey,
true);
1166 fragmentShader <<
" float qt_heightAmount = qt_material_properties4.x;\n";
1167 maskVariableByVertexColorChannel(
"qt_heightAmount", QSSGRenderDefaultMaterial::HeightAmountMask );
1168 fragmentShader <<
" qt_texCoord0 = qt_parallaxMapping(" << samplerState.fragCoordsName(QSSGRenderableImage::Type::Height) <<
",\n"
1169 <<
" " << samplerState.samplerName(QSSGRenderableImage::Type::Height) <<
",\n"
1171 <<
" qt_binormal,\n"
1172 <<
" qt_world_normal,\n"
1173 <<
" qt_varWorldPos, \n"
1174 <<
"#if QSHADER_VIEW_COUNT >= 2\n"
1175 <<
" qt_cameraPosition[qt_viewIndex],\n"
1177 <<
" qt_cameraPosition,\n"
1179 <<
" qt_heightAmount,\n"
1180 <<
" qt_material_properties4.y,\n"
1181 <<
" qt_material_properties4.z);\n";
1186 addLocalVariable(fragmentShader,
"qt_clearcoatNormal",
"vec3");
1191 if (hasCustomFrag) {
1192 fragmentShader <<
" qt_clearcoatNormal = qt_customClearcoatNormal;\n";
1194 if (samplerState.hasImage(QSSGRenderableImage::Type::ClearcoatNormal)) {
1195 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::ClearcoatNormal, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1196 fragmentShader.addFunction(
"sampleNormalTexture");
1197 fragmentShader <<
" float qt_clearcoat_normal_strength = qt_material_clearcoat_normal_strength;\n";
1198 maskVariableByVertexColorChannel(
"qt_clearcoat_normal_strength", QSSGRenderDefaultMaterial::ClearcoatNormalStrengthMask );
1199 fragmentShader <<
" qt_clearcoatNormal = qt_sampleNormalTexture(" << samplerState.samplerName(QSSGRenderableImage::Type::ClearcoatNormal)
1200 <<
", qt_clearcoat_normal_strength, "
1201 << samplerState.fragCoordsName(QSSGRenderableImage::Type::ClearcoatNormal)
1202 <<
", qt_tangent, qt_binormal, qt_world_normal);\n";
1206 fragmentShader <<
" qt_clearcoatNormal = qt_world_normal;\n";
1213 if (samplerState.hasImage(QSSGRenderableImage::Type::Bump)) {
1214 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Bump, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1215 fragmentShader.append(
" float qt_bumpAmount = qt_material_properties2.y;\n");
1216 maskVariableByVertexColorChannel(
"qt_bumpAmount", QSSGRenderDefaultMaterial::NormalStrengthMask );
1217 fragmentShader.addInclude(
"defaultMaterialBumpNoLod.glsllib");
1218 fragmentShader <<
" qt_world_normal = qt_defaultMaterialBumpNoLod("
1219 << samplerState.samplerName(QSSGRenderableImage::Type::Bump)
1220 <<
", qt_bumpAmount, " << samplerState.fragCoordsName(QSSGRenderableImage::Type::Bump)
1221 <<
", qt_tangent, qt_binormal, qt_world_normal);\n";
1222 }
else if (samplerState.hasImage(QSSGRenderableImage::Type::Normal)) {
1223 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Normal, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1224 fragmentShader.append(
" float qt_normalStrength = qt_material_properties2.y;\n");
1225 maskVariableByVertexColorChannel(
"qt_normalStrength", QSSGRenderDefaultMaterial::NormalStrengthMask );
1226 fragmentShader.addFunction(
"sampleNormalTexture");
1227 fragmentShader <<
" qt_world_normal = qt_sampleNormalTexture(" << samplerState.samplerName(QSSGRenderableImage::Type::Normal)
1228 <<
", qt_normalStrength, " << samplerState.fragCoordsName(QSSGRenderableImage::Type::Normal)
1229 <<
", qt_tangent, qt_binormal, qt_world_normal);\n";
1237 fragmentShader.append(
" vec3 tmp_light_color;");
1241 fragmentShader.append(
" vec3 qt_specularBase;");
1242 fragmentShader.addUniform(
"qt_material_specular",
"vec4");
1244 fragmentShader.append(
" vec3 qt_specularTint = vec3(1.0);");
1246 fragmentShader.append(
" vec3 qt_specularTint = qt_material_specular.rgb;");
1250 if ((samplerState.hasImage(QSSGRenderableImage::Type::BaseColor) || samplerState.hasImage(QSSGRenderableImage::Type::Diffuse)) && passRequirmentState
.needsBaseColor) {
1253 QSSGRenderableImage::Type baseImageType = QSSGRenderableImage::Type::Unknown;
1254 if (samplerState.hasImage(QSSGRenderableImage::Type::BaseColor))
1255 baseImageType = QSSGRenderableImage::Type::BaseColor;
1256 else if (samplerState.hasImage(QSSGRenderableImage::Type::Diffuse))
1257 baseImageType = QSSGRenderableImage::Type::Diffuse;
1260 samplerState.generateImageUVAndSampler(baseImageType, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1262 if (keyProps.m_baseColorSingleChannelEnabled.getValue(inKey)) {
1263 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::BaseColorChannel];
1264 fragmentShader <<
" vec4 qt_base_texture_color = vec4(vec3(texture2D(" << samplerState.samplerName(baseImageType)
1265 <<
", " << samplerState.fragCoordsName(baseImageType) <<
")" << channelStr(channelProps, inKey) <<
"), 1.0f);\n";
1267 fragmentShader <<
" vec4 qt_base_texture_color = texture2D(" << samplerState.samplerName(baseImageType)
1268 <<
", " << samplerState.fragCoordsName(baseImageType) <<
");\n";
1271 if (keyProps.m_imageMaps[QSSGShaderDefaultMaterialKeyProperties::BaseColorMap].isPreMultipliedAlpha(inKey))
1272 fragmentShader <<
" qt_base_texture_color.rgb /= qt_base_texture_color.a;\n";
1274 if (!keyProps.m_imageMaps[QSSGShaderDefaultMaterialKeyProperties::BaseColorMap].isLinear(inKey)) {
1276 fragmentShader.addInclude(
"tonemapping.glsllib");
1277 fragmentShader <<
" qt_base_texture_color = qt_sRGBToLinear(qt_base_texture_color);\n";
1280 fragmentShader <<
" qt_diffuseColor *= qt_base_texture_color;\n";
1284 if (keyProps.m_alphaMode.getAlphaMode(inKey) == QSSGRenderDefaultMaterial::MaterialAlphaMode::Mask) {
1288 fragmentShader <<
" if (qt_diffuseColor.a < qt_material_properties3.y) {\n"
1289 <<
" qt_diffuseColor = vec4(0.0);\n"
1292 <<
" qt_diffuseColor.a = 1.0;\n"
1294 }
else if (keyProps.m_alphaMode.getAlphaMode(inKey) == QSSGRenderDefaultMaterial::MaterialAlphaMode::Opaque) {
1295 fragmentShader <<
" qt_diffuseColor.a = 1.0;\n";
1299 if (samplerState.hasImage(QSSGRenderableImage::Type::Opacity)) {
1300 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Opacity, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1301 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::OpacityChannel];
1302 fragmentShader <<
" float qt_opacity_map_value = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Opacity)
1303 <<
", " << samplerState.fragCoordsName(QSSGRenderableImage::Type::Opacity) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1304 if (keyProps.m_invertOpacityMapValue.getValue(inKey))
1305 fragmentShader <<
" qt_opacity_map_value = 1.0 - qt_opacity_map_value;\n";
1306 fragmentShader <<
" qt_objectOpacity *= qt_opacity_map_value;\n";
1311 addLocalVariable(fragmentShader,
"qt_aoFactor",
"float");
1314 fragmentShader.addInclude(
"ssao.glsllib");
1315 fragmentShader.append(
" qt_aoFactor = qt_screenSpaceAmbientOcclusionFactor();");
1317 fragmentShader.append(
" qt_aoFactor = 1.0;");
1321 fragmentShader <<
" qt_aoFactor *= qt_customOcclusionAmount;\n";
1327 fragmentShader <<
" float qt_roughnessAmount = qt_customSpecularRoughness;\n";
1329 fragmentShader <<
" float qt_roughnessAmount = qt_material_properties.y;\n";
1331 maskVariableByVertexColorChannel(
"qt_roughnessAmount", QSSGRenderDefaultMaterial::RoughnessMask );
1333 if (samplerState.hasImage(QSSGRenderableImage::Type::Roughness)) {
1334 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Roughness, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1335 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::RoughnessChannel];
1336 fragmentShader <<
" qt_roughnessAmount *= texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Roughness) <<
", "
1337 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Roughness) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1342 fragmentShader <<
" qt_roughnessAmount = clamp(1.0 - qt_roughnessAmount, 0.0, 1.0);\n";
1348 fragmentShader <<
" float qt_metalnessAmount = qt_customMetalnessAmount;\n";
1350 fragmentShader <<
" float qt_metalnessAmount = qt_material_properties.z;\n";
1352 fragmentShader <<
" float qt_metalnessAmount = 0.0;\n";
1354 maskVariableByVertexColorChannel(
"qt_metalnessAmount", QSSGRenderDefaultMaterial::MetalnessMask );
1356 if (passRequirmentState
.hasSpecularLight && samplerState.hasImage(QSSGRenderableImage::Type::Metalness)) {
1357 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::MetalnessChannel];
1358 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Metalness, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1359 fragmentShader <<
" float qt_sampledMetalness = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Metalness) <<
", "
1360 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Metalness) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1361 fragmentShader <<
" qt_metalnessAmount = clamp(qt_metalnessAmount * qt_sampledMetalness, 0.0, 1.0);\n";
1367 fragmentShader <<
" if ((qt_diffuseColor.a * qt_objectOpacity) < 1.0)\n";
1368 fragmentShader <<
" discard;\n";
1374 vertexShader.generateViewVector(inKey);
1375 fragmentShader.addUniform(
"qt_material_properties",
"vec4");
1378 fragmentShader <<
" qt_specularBase = vec3(1.0);\n";
1380 fragmentShader <<
" qt_specularBase = qt_diffuseColor.rgb;\n";
1382 fragmentShader <<
" float qt_specularFactor = qt_customSpecularAmount;\n";
1384 fragmentShader <<
" float qt_specularFactor = qt_material_properties.x;\n";
1386 maskVariableByVertexColorChannel(
"qt_specularFactor", QSSGRenderDefaultMaterial::SpecularAmountMask );
1389 fragmentShader.addUniform(
"qt_light_ambient_total",
"vec3");
1391 fragmentShader.append(
" vec4 global_diffuse_light = vec4(0.0);");
1394 fragmentShader <<
" global_diffuse_light.rgb = qt_lightmap_color(qt_texCoordLightmap) * (1.0 - qt_metalnessAmount) * qt_diffuseColor.rgb;\n";
1396 if (hasCustomFrag && hasCustomFunction(QByteArrayLiteral(
"qt_ambientLightProcessor"))) {
1398 fragmentShader.append(
" qt_ambientLightProcessor(global_diffuse_light.rgb, qt_light_ambient_total.rgb * (1.0 - qt_metalnessAmount) * qt_diffuseColor.rgb, qt_world_normal, qt_view_vector");
1400 fragmentShader <<
", qt_customShared);\n";
1402 fragmentShader <<
");\n";
1404 fragmentShader.append(
" global_diffuse_light = vec4(qt_light_ambient_total.rgb * (1.0 - qt_metalnessAmount) * qt_diffuseColor.rgb, 0.0);");
1408 fragmentShader.append(
" vec3 global_specular_light = vec3(0.0);");
1412 if (samplerState.hasImage(QSSGRenderableImage::Type::SpecularAmountMap)) {
1413 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::SpecularAmountMap, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1415 if (keyProps.m_specularSingleChannelEnabled.getValue(inKey)) {
1416 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::SpecularAmountChannel];
1417 fragmentShader <<
" vec4 qt_specular_amount_map = vec4(vec3(texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::SpecularAmountMap)
1418 <<
", " << samplerState.fragCoordsName(QSSGRenderableImage::Type::SpecularAmountMap) <<
")" << channelStr(channelProps, inKey) <<
"), 1.0f);\n";
1420 fragmentShader <<
" vec4 qt_specular_amount_map = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::SpecularAmountMap)
1421 <<
", " << samplerState.fragCoordsName(QSSGRenderableImage::Type::SpecularAmountMap) <<
");\n";
1423 fragmentShader <<
" qt_specularBase *= qt_sRGBToLinear(qt_specular_amount_map).rgb;\n";
1428 fragmentShader <<
" qt_specularTint *= qt_specularBase;\n";
1429 fragmentShader <<
" vec3 qt_specularAmount = vec3(1.0);\n";
1431 fragmentShader <<
" vec3 qt_specularAmount = qt_specularBase * vec3(qt_metalnessAmount + qt_specularFactor * (1.0 - qt_metalnessAmount));\n";
1435 if (samplerState.hasImage(QSSGRenderableImage::Type::Translucency)) {
1436 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Translucency, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1437 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::TranslucencyChannel];
1438 fragmentShader <<
" float qt_translucent_depth_range = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Translucency)
1439 <<
", " << samplerState.fragCoordsName(QSSGRenderableImage::Type::Translucency) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1440 fragmentShader <<
" float qt_translucent_thickness = qt_translucent_depth_range * qt_translucent_depth_range;\n";
1441 fragmentShader <<
" float qt_translucent_thickness_exp = exp(qt_translucent_thickness * qt_material_properties2.z);\n";
1445 if (samplerState.hasImage(QSSGRenderableImage::Type::Occlusion)) {
1446 addLocalVariable(fragmentShader,
"qt_ao",
"float");
1447 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Occlusion, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1448 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::OcclusionChannel];
1449 fragmentShader <<
" qt_ao = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Occlusion) <<
", "
1450 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Occlusion) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1451 fragmentShader <<
" qt_aoFactor *= qt_ao * qt_material_properties3.x;\n";
1452 maskVariableByVertexColorChannel(
"qt_aoFactor", QSSGRenderDefaultMaterial::OcclusionAmountMask );
1456 addLocalVariable(fragmentShader,
"qt_clearcoatAmount",
"float");
1457 addLocalVariable(fragmentShader,
"qt_clearcoatRoughness",
"float");
1458 addLocalVariable(fragmentShader,
"qt_clearcoatF0",
"vec3");
1459 addLocalVariable(fragmentShader,
"qt_clearcoatF90",
"vec3");
1460 addLocalVariable(fragmentShader,
"qt_global_clearcoat",
"vec3");
1463 fragmentShader <<
" qt_clearcoatAmount = qt_customClearcoatAmount;\n";
1465 fragmentShader <<
" qt_clearcoatAmount = qt_material_properties3.z;\n";
1466 maskVariableByVertexColorChannel(
"qt_clearcoatAmount", QSSGRenderDefaultMaterial::ClearcoatAmountMask );
1468 fragmentShader <<
" qt_clearcoatRoughness = qt_customClearcoatRoughness;\n";
1470 fragmentShader <<
" qt_clearcoatRoughness = qt_material_properties3.w;\n";
1471 maskVariableByVertexColorChannel(
"qt_clearcoatRoughness", QSSGRenderDefaultMaterial::ClearcoatRoughnessAmountMask );
1472 fragmentShader <<
" qt_clearcoatF0 = vec3(((1.0-qt_iOR) * (1.0-qt_iOR)) / ((1.0+qt_iOR) * (1.0+qt_iOR)));\n";
1473 fragmentShader <<
" qt_clearcoatF90 = vec3(1.0);\n";
1474 fragmentShader <<
" qt_global_clearcoat = vec3(0.0);\n";
1476 if (samplerState.hasImage(QSSGRenderableImage::Type::Clearcoat)) {
1477 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Clearcoat, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1478 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::ClearcoatChannel];
1479 fragmentShader <<
" qt_clearcoatAmount *= texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Clearcoat) <<
", "
1480 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Clearcoat) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1483 if (samplerState.hasImage(QSSGRenderableImage::Type::ClearcoatRoughness)) {
1484 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::ClearcoatRoughness, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1485 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::ClearcoatRoughnessChannel];
1486 fragmentShader <<
" qt_clearcoatRoughness *= texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::ClearcoatRoughness) <<
", "
1487 << samplerState.fragCoordsName(QSSGRenderableImage::Type::ClearcoatRoughness) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1488 fragmentShader <<
" qt_clearcoatRoughness = clamp(qt_clearcoatRoughness, 0.0, 1.0);\n";
1493 fragmentShader.addInclude(
"transmission.glsllib");
1494 addLocalVariable(fragmentShader,
"qt_transmissionFactor",
"float");
1495 addLocalVariable(fragmentShader,
"qt_global_transmission",
"vec3");
1497 addLocalVariable(fragmentShader,
"qt_thicknessFactor",
"float");
1498 addLocalVariable(fragmentShader,
"qt_attenuationColor",
"vec3");
1499 addLocalVariable(fragmentShader,
"qt_attenuationDistance",
"float");
1500 fragmentShader <<
" qt_global_transmission = vec3(0.0);\n";
1502 if (hasCustomFrag) {
1503 fragmentShader <<
" qt_transmissionFactor = qt_customTransmissionFactor;\n";
1504 fragmentShader <<
" qt_thicknessFactor = qt_customThicknessFactor;\n";
1505 fragmentShader <<
" qt_attenuationColor = qt_customAttenuationColor;\n";
1506 fragmentShader <<
" qt_attenuationDistance = qt_customAttenuationDistance;\n";
1508 fragmentShader <<
" qt_transmissionFactor = qt_material_properties4.w;\n";
1509 maskVariableByVertexColorChannel(
"qt_transmissionFactor", QSSGRenderDefaultMaterial::TransmissionFactorMask );
1511 if (samplerState.hasImage(QSSGRenderableImage::Type::Transmission)) {
1512 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Transmission, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1513 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::TransmissionChannel];
1514 fragmentShader <<
" qt_transmissionFactor *= texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Transmission) <<
", "
1515 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Transmission) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1518 fragmentShader <<
" qt_thicknessFactor = qt_material_thickness;\n";
1519 maskVariableByVertexColorChannel(
"qt_thicknessFactor", QSSGRenderDefaultMaterial::ThicknessFactorMask );
1520 fragmentShader <<
" qt_attenuationColor = qt_material_attenuation.xyz;\n";
1521 fragmentShader <<
" qt_attenuationDistance = qt_material_attenuation.w;\n";
1523 if (samplerState.hasImage(QSSGRenderableImage::Type::Thickness)) {
1524 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Thickness, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1525 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::ThicknessChannel];
1526 fragmentShader <<
" qt_thicknessFactor *= texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Thickness) <<
", "
1527 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Thickness) <<
")" << channelStr(channelProps, inKey) <<
";\n";
1532 fragmentShader <<
" vec3 qt_f0 = vec3(1.0);\n";
1533 fragmentShader <<
" vec3 qt_f90 = vec3(1.0);\n";
1537 fragmentShader.addInclude(
"principledMaterialFresnel.glsllib");
1539 fragmentShader <<
" qt_f0 = qt_F0_ior(qt_iOR, qt_metalnessAmount, qt_diffuseColor.rgb);\n";
1541 fragmentShader <<
" const float qt_reflectance = max(max(qt_specularTint.r, qt_specularTint.g), qt_specularTint.b);\n";
1542 fragmentShader <<
" qt_f0 = qt_specularTint;\n";
1543 fragmentShader <<
" qt_specularTint = vec3(1.0);\n";
1544 fragmentShader <<
" qt_f90 = vec3(clamp(qt_reflectance * 50.0, 0.0, 1.0));\n";
1545 fragmentShader <<
" qt_diffuseColor.rgb *= (1 - qt_reflectance);\n";
1549 fragmentShader.append(
" vec3 vNormalWsDdx = dFdx(qt_world_normal.xyz);\n");
1550 fragmentShader.append(
" vec3 vNormalWsDdy = dFdy(qt_world_normal.xyz);\n");
1551 fragmentShader.append(
" float flGeometricRoughnessFactor = pow(clamp(max(dot(vNormalWsDdx, vNormalWsDdx), dot(vNormalWsDdy, vNormalWsDdy)), 0.0, 1.0), 0.333);\n");
1552 fragmentShader.append(
" qt_roughnessAmount = max(flGeometricRoughnessFactor, qt_roughnessAmount);\n");
1556 fragmentShader <<
" float qt_fresnelPower = qt_customFresnelPower;\n";
1558 fragmentShader <<
" float qt_fresnelPower = qt_material_properties2.x;\n";
1561 fragmentShader <<
" vec3 qt_principledMaterialFresnelValue = qt_principledMaterialFresnel(qt_world_normal, qt_view_vector, qt_f0, qt_roughnessAmount, qt_fresnelPower);\n";
1563 if (hasCustomFrag) {
1564 fragmentShader <<
" float qt_fresnelScale = qt_customFresnelScale;\n";
1565 fragmentShader <<
" float qt_fresnelBias = qt_customFresnelBias;\n";
1567 fragmentShader <<
" float qt_fresnelScale = qt_material_properties5.x;\n";
1568 fragmentShader <<
" float qt_fresnelBias = qt_material_properties5.y;\n";
1570 fragmentShader <<
" qt_principledMaterialFresnelValue = clamp(vec3(qt_fresnelBias) + "
1571 <<
"qt_fresnelScale * qt_principledMaterialFresnelValue, 0.0, 1.0);\n";
1573 fragmentShader <<
" qt_specularAmount *= qt_principledMaterialFresnelValue;\n";
1577 fragmentShader <<
" qt_specularTint = mix(vec3(1.0), qt_specularTint, 1.0 - qt_metalnessAmount);\n";
1580 fragmentShader <<
" qt_specularAmount *= qt_principledMaterialFresnel(qt_world_normal, qt_view_vector, qt_f0, qt_roughnessAmount, qt_fresnelPower);\n";
1585 fragmentShader.addUniform(
"qt_lightAndShadowCounts",
"vec4");
1586 fragmentShader.addFunction(
"processPunctualLighting");
1587 fragmentShader <<
" qt_processPunctualLighting(global_diffuse_light.rgb,\n"
1588 <<
" global_specular_light.rgb,\n"
1589 <<
" qt_diffuseColor.rgb,\n"
1590 <<
" qt_varWorldPos,\n"
1591 <<
" qt_world_normal.xyz,\n"
1592 <<
" qt_view_vector,\n"
1593 <<
"#if QSSG_ENABLE_SPECULAR\n"
1594 <<
" qt_specularAmount,\n"
1595 <<
" qt_specularTint,\n"
1596 <<
"#endif // QSSG_ENABLE_SPECULAR\n"
1597 <<
" qt_roughnessAmount,\n"
1598 <<
" qt_metalnessAmount,\n"
1599 <<
"#if QSSG_CUSTOM_MATERIAL_DIRECTIONAL_LIGHT_PROCESSOR || QSSG_CUSTOM_MATERIAL_POINT_LIGHT_PROCESSOR || QSSG_CUSTOM_MATERIAL_SPOT_LIGHT_PROCESSOR || QSSG_CUSTOM_MATERIAL_SPECULAR_PROCESSOR\n"
1600 <<
" qt_customBaseColor,\n"
1601 <<
"#endif // QSSG_CUSTOM_MATERIAL_*\n"
1602 <<
"#if QSSG_CUSTOM_MATERIAL_SPECULAR_PROCESSOR\n"
1603 <<
" qt_customSpecularAmount,\n"
1604 <<
"#endif // QSSG_CUSTOM_MATERIAL_SPECULAR_PROCESSOR\n"
1605 <<
"#if QSSG_CUSTOM_MATERIAL_SHARED_VARIABLES\n"
1606 <<
" qt_customShared,\n"
1607 <<
"#endif // QSSG_CUSTOM_MATERIAL_SHARED_VARIABLES\n"
1608 <<
"#if QSSG_ENABLE_CLEARCOAT\n"
1609 <<
" qt_global_clearcoat,\n"
1610 <<
" qt_clearcoatNormal,\n"
1611 <<
" qt_clearcoatRoughness,\n"
1612 <<
" qt_clearcoatF0,\n"
1613 <<
" qt_clearcoatF90,\n"
1614 <<
"#endif // QSSG_ENABLE_CLEARCOAT\n"
1615 <<
"#if QSSG_ENABLE_TRANSMISSION\n"
1616 <<
" qt_global_transmission,\n"
1617 <<
" qt_thicknessFactor,\n"
1619 <<
" qt_transmissionFactor,\n"
1620 <<
" qt_attenuationColor,\n"
1621 <<
" qt_attenuationDistance,\n"
1622 <<
"#endif // QSSG_ENABLE_TRANSMISSION\n"
1630 fragmentShader <<
" global_diffuse_light = vec4(global_diffuse_light.rgb * qt_aoFactor, qt_objectOpacity * qt_diffuseColor.a);\n";
1633 vertexShader.generateWorldNormal(inKey);
1634 fragmentShader.addInclude(
"sampleReflectionProbe.glsllib");
1638 fragmentShader <<
" global_diffuse_light.rgb += qt_diffuseColor.rgb * (1.0 - qt_specularAmount) * qt_sampleDiffuseReflection(qt_reflectionMap, qt_world_normal).rgb;\n";
1640 fragmentShader <<
" global_diffuse_light.rgb += qt_diffuseColor.rgb * qt_sampleDiffuseReflection(qt_reflectionMap, qt_world_normal).rgb;\n";
1645 fragmentShader <<
" global_specular_light += qt_specularTint * qt_sampleGlossyReflectionPrincipled(qt_reflectionMap, qt_world_normal, qt_view_vector, qt_specularAmount, qt_roughnessAmount).rgb;\n";
1647 fragmentShader <<
" global_specular_light += qt_specularAmount * qt_specularTint * qt_sampleGlossyReflection(qt_reflectionMap, qt_world_normal, qt_view_vector, qt_roughnessAmount).rgb;\n";
1652 fragmentShader <<
" qt_global_clearcoat += qt_sampleGlossyReflectionPrincipled(qt_reflectionMap, qt_clearcoatNormal, qt_view_vector, qt_clearcoatF0, qt_clearcoatRoughness).rgb;\n";
1655 vertexShader.generateWorldNormal(inKey);
1656 fragmentShader.addInclude(
"sampleProbe.glsllib");
1657 if (hasCustomIblProbe) {
1659 fragmentShader <<
" vec3 qt_iblDiffuse = vec3(0.0);\n";
1660 fragmentShader <<
" vec3 qt_iblSpecular = vec3(0.0);\n";
1661 fragmentShader <<
" qt_iblProbeProcessor(qt_iblDiffuse, qt_iblSpecular, qt_customBaseColor, qt_aoFactor, qt_specularFactor, qt_roughnessAmount, qt_world_normal, qt_view_vector";
1663 fragmentShader <<
", qt_lightProbeOrientation";
1665 fragmentShader <<
", mat3(1.0)";
1667 fragmentShader <<
", qt_customShared);\n";
1669 fragmentShader <<
");\n";
1673 fragmentShader <<
" vec3 qt_iblDiffuse = qt_diffuseColor.rgb * (1.0 - qt_specularAmount) * qt_sampleDiffuse(qt_world_normal).rgb;\n";
1675 fragmentShader <<
" vec3 qt_iblDiffuse = qt_diffuseColor.rgb * qt_sampleDiffuse(qt_world_normal).rgb;\n";
1680 fragmentShader <<
" vec3 qt_iblSpecular = qt_specularTint * qt_sampleGlossyPrincipled(qt_world_normal, qt_view_vector, qt_specularAmount, qt_roughnessAmount).rgb;\n";
1682 fragmentShader <<
" vec3 qt_iblSpecular = qt_specularAmount * qt_specularTint * qt_sampleGlossy(qt_world_normal, qt_view_vector, qt_roughnessAmount).rgb;\n";
1687 fragmentShader <<
" vec3 qt_iblClearcoat = qt_sampleGlossyPrincipled(qt_clearcoatNormal, qt_view_vector, qt_clearcoatF0, qt_clearcoatRoughness).rgb;\n";
1690 fragmentShader <<
" global_diffuse_light.rgb += qt_iblDiffuse * qt_aoFactor;\n";
1692 fragmentShader <<
" global_specular_light += qt_iblSpecular * qt_aoFactor;\n";
1694 fragmentShader <<
" qt_global_clearcoat += qt_iblClearcoat * qt_aoFactor;\n";
1695 }
else if (hasCustomIblProbe) {
1697 fragmentShader.addUniform(
"qt_lightProbe",
"samplerCube");
1698 fragmentShader.addUniform(
"qt_lightProbeProperties",
"vec4");
1703 fragmentShader <<
" qt_global_transmission += qt_transmissionFactor * qt_getIBLVolumeRefraction(qt_world_normal, qt_view_vector, qt_roughnessAmount, "
1704 "qt_diffuseColor.rgb, qt_specularAmount, qt_varWorldPos, qt_iOR, qt_thicknessFactor, qt_attenuationColor, qt_attenuationDistance);\n";
1709 if (samplerState.hasImage(QSSGRenderableImage::Type::Specular) || samplerState.hasImage(QSSGRenderableImage::Type::Emissive)) {
1710 addLocalVariable(fragmentShader,
"qt_texture_color",
"vec4");
1712 if (samplerState.hasImage(QSSGRenderableImage::Type::Specular)) {
1713 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Specular, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1714 fragmentShader <<
" qt_texture_color = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Specular) <<
", "
1715 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Specular) <<
");\n";
1716 fragmentShader.addInclude(
"tonemapping.glsllib");
1717 fragmentShader <<
" global_specular_light += qt_sRGBToLinear(qt_texture_color.rgb) * qt_specularTint;\n";
1718 fragmentShader <<
" global_diffuse_light.a *= qt_texture_color.a;\n";
1721 if (samplerState.hasImage(QSSGRenderableImage::Type::Emissive)) {
1722 samplerState.generateImageUVAndSampler(QSSGRenderableImage::Type::Emissive, vertexShader, fragmentShader, inKey, passRequirmentState
.hasParallaxMapping);
1723 fragmentShader <<
" qt_texture_color = texture2D(" << samplerState.samplerName(QSSGRenderableImage::Type::Emissive) <<
", "
1724 << samplerState.fragCoordsName(QSSGRenderableImage::Type::Emissive) <<
");\n";
1725 fragmentShader.addInclude(
"tonemapping.glsllib");
1726 if (keyProps.m_emissiveSingleChannelEnabled.getValue(inKey)) {
1727 const auto &channelProps = keyProps.m_textureChannels[QSSGShaderDefaultMaterialKeyProperties::EmissiveChannel];
1728 fragmentShader <<
" qt_global_emission *= qt_sRGBToLinear(vec3(qt_texture_color" <<
1729 channelStr(channelProps, inKey) <<
"));\n";
1731 fragmentShader <<
" qt_global_emission *= qt_sRGBToLinear(qt_texture_color.rgb);\n";
1737 fragmentShader <<
" global_diffuse_light.rgb = mix(global_diffuse_light.rgb, qt_global_transmission, qt_transmissionFactor);\n";
1740 fragmentShader <<
" global_diffuse_light.rgb *= 1.0 - qt_metalnessAmount;\n";
1743 if (passRequirmentState
.hasFog) {
1744 fragmentShader.addInclude(
"fog.glsllib");
1745 fragmentShader <<
" calculateFog(qt_global_emission, global_specular_light, global_diffuse_light.rgb);\n";
1748 fragmentShader <<
" vec4 qt_color_sum = vec4(global_diffuse_light.rgb + global_specular_light + qt_global_emission, global_diffuse_light.a);\n";
1751 fragmentShader.addInclude(
"bsdf.glsllib");
1753 fragmentShader <<
" float qt_clearcoatFresnelPower = qt_customClearcoatFresnelPower;\n";
1755 fragmentShader <<
" float qt_clearcoatFresnelPower = qt_material_clearcoat_fresnel_power;\n";
1756 fragmentShader <<
" vec3 qt_clearcoatFresnel = qt_schlick3(qt_clearcoatF0, qt_clearcoatF90, clamp(dot(qt_clearcoatNormal, qt_view_vector), 0.0, 1.0), qt_clearcoatFresnelPower);\n";
1758 if (hasCustomFrag) {
1759 fragmentShader <<
" float qt_clearcoatFresnelScale = qt_customClearcoatFresnelScale;\n";
1760 fragmentShader <<
" float qt_clearcoatFresnelBias = qt_customClearcoatFresnelBias;\n";
1762 fragmentShader <<
" float qt_clearcoatFresnelScale = qt_material_properties5.z;\n";
1763 fragmentShader <<
" float qt_clearcoatFresnelBias = qt_material_properties5.w;\n";
1765 fragmentShader <<
" qt_clearcoatFresnel = clamp(vec3(qt_clearcoatFresnelBias) + qt_clearcoatFresnelScale * qt_clearcoatFresnel, 0.0, 1.0);\n";
1767 fragmentShader <<
" qt_global_clearcoat = qt_global_clearcoat * qt_clearcoatAmount;\n";
1768 fragmentShader <<
" qt_color_sum.rgb = qt_color_sum.rgb * (1.0 - qt_clearcoatAmount * qt_clearcoatFresnel) + qt_global_clearcoat;\n";
1771 if (hasCustomFrag && hasCustomFunction(QByteArrayLiteral(
"qt_customPostProcessor"))) {
1773 fragmentShader <<
" qt_customPostProcessor(qt_color_sum, global_diffuse_light, global_specular_light, qt_global_emission, qt_texCoord0, qt_texCoord1";
1775 fragmentShader <<
", qt_customShared);\n";
1777 fragmentShader <<
");\n";
1789 fragmentShader.append(
" vec4 qt_color_sum = vec4(qt_diffuseColor.rgb, qt_diffuseColor.a * qt_objectOpacity);");
1791 if (passRequirmentState.oitMethod == QSSGRenderLayer::OITMethod::WeightedBlended) {
1792 fragmentShader.addInclude(
"oitweightedblended.glsllib");
1793 fragmentShader.addInclude(
"tonemapping.glsllib");
1794 fragmentShader.addUniform(
"qt_cameraPosition",
"vec3");
1795 fragmentShader.addUniform(
"qt_cameraProperties",
"vec2");
1796 fragmentShader.append(
" float z = abs(gl_FragCoord.z);");
1797 fragmentShader.append(
" qt_color_sum.rgb = qt_tonemap(qt_color_sum.rgb) * qt_color_sum.a;");
1798 fragmentShader.append(
" fragOutput = qt_color_sum * qt_transparencyWeight(z, qt_color_sum.a, qt_cameraProperties.y);");
1799 fragmentShader.append(
" revealageOutput = vec4(qt_color_sum.a);");
1800 }
else if (passRequirmentState.oitMethod == QSSGRenderLayer::OITMethod::LinkedList) {
1801 fragmentShader.addInclude(
"tonemapping.glsllib");
1802 fragmentShader.addUniform(
"qt_listNodeCount",
"uint");
1803 fragmentShader.addUniform(
"qt_ABufImageWidth",
"uint");
1804 fragmentShader.addUniform(
"qt_viewSize",
"ivec2");
1806 fragmentShader.addDefinition(
"QSSG_MULTISAMPLE",
"1");
1807#ifdef QSSG_OIT_USE_BUFFERS
1808 QSSGShaderResourceMergeContext::setAdditionalBufferAmount(3);
1809 fragmentShader.addUniform(
"qt_samples",
"uint");
1810 fragmentShader.addInclude(
"oitlinkedlist_buf.glsllib");
1812 fragmentShader.append(
" fragOutput = qt_oitLinkedList(qt_tonemap(qt_color_sum), qt_listNodeCount, qt_ABufImageWidth, qt_viewSize, qt_viewIndex, qt_samples);");
1814 fragmentShader.append(
" fragOutput = qt_oitLinkedList(qt_tonemap(qt_color_sum), qt_listNodeCount, qt_ABufImageWidth, qt_viewSize, 0, qt_samples);");
1816 fragmentShader.addInclude(
"oitlinkedlist.glsllib");
1818 fragmentShader.append(
" fragOutput = qt_oitLinkedList(qt_tonemap(qt_color_sum), qt_listNodeCount, qt_ABufImageWidth, qt_viewSize, qt_viewIndex);");
1820 fragmentShader.append(
" fragOutput = qt_oitLinkedList(qt_tonemap(qt_color_sum), qt_listNodeCount, qt_ABufImageWidth, qt_viewSize, 0);");
1824 fragmentShader.addInclude(
"tonemapping.glsllib");
1825 fragmentShader.append(
" fragOutput = vec4(qt_tonemap(qt_color_sum));");
1829 fragmentShader <<
" vec4 fragOutput = vec4(0.0);\n";
1832 Q_ASSERT(viewCount == 1);
1833 fragmentShader <<
" // directional shadow pass\n"
1834 <<
" float qt_shadowDepth = (qt_varDepth + qt_shadowDepthAdjust.x) * qt_shadowDepthAdjust.y;\n"
1835 <<
" fragOutput = vec4(qt_shadowDepth);\n";
1838 Q_ASSERT(viewCount == 1);
1839 fragmentShader.addUniform(
"qt_cameraPosition",
"vec3");
1840 fragmentShader.addUniform(
"qt_cameraProperties",
"vec2");
1841 fragmentShader <<
" // omnidirectional shadow pass\n"
1842 <<
" vec3 qt_shadowCamPos = vec3(qt_cameraPosition.x, qt_cameraPosition.y, qt_cameraPosition.z);\n"
1843 <<
" float qt_shadowDist = length(qt_varShadowWorldPos - qt_shadowCamPos);\n"
1844 <<
" qt_shadowDist = (qt_shadowDist - qt_cameraProperties.x) / (qt_cameraProperties.y - qt_cameraProperties.x);\n"
1845 <<
" fragOutput = vec4(qt_shadowDist, qt_shadowDist, qt_shadowDist, 1.0);\n";
1849 fragmentShader.append(
" fragOutput = vec4(qt_world_normal, qt_roughnessAmount);\n");
1852 fragmentShader.append(
" vec3 debugOutput = vec3(0.0);\n");
1853 switch (passRequirmentState.debugMode) {
1854 case QSSGRenderLayer::MaterialDebugMode::BaseColor:
1855 fragmentShader.addInclude(
"tonemapping.glsllib");
1856 fragmentShader.append(
" debugOutput += qt_tonemap(qt_diffuseColor.rgb);\n");
1858 case QSSGRenderLayer::MaterialDebugMode::Roughness:
1859 fragmentShader.append(
" debugOutput += vec3(qt_roughnessAmount);\n");
1861 case QSSGRenderLayer::MaterialDebugMode::Metalness:
1862 fragmentShader.append(
" debugOutput += vec3(qt_metalnessAmount);\n");
1864 case QSSGRenderLayer::MaterialDebugMode::Diffuse:
1865 fragmentShader.addInclude(
"tonemapping.glsllib");
1866 fragmentShader.append(
" debugOutput += qt_tonemap(global_diffuse_light.rgb);\n");
1868 case QSSGRenderLayer::MaterialDebugMode::Specular:
1869 fragmentShader.addInclude(
"tonemapping.glsllib");
1870 fragmentShader.append(
" debugOutput += qt_tonemap(global_specular_light);\n");
1872 case QSSGRenderLayer::MaterialDebugMode::ShadowOcclusion:
1875 fragmentShader.addFunction(
"debugShadowOcclusion");
1876 vertexShader.generateWorldPosition(inKey);
1877 fragmentShader.append(
" debugOutput += vec3(qt_debugShadowOcclusion());\n");
1879 case QSSGRenderLayer::MaterialDebugMode::Emission:
1880 fragmentShader.addInclude(
"tonemapping.glsllib");
1881 fragmentShader.append(
" debugOutput += qt_tonemap(qt_global_emission);\n");
1883 case QSSGRenderLayer::MaterialDebugMode::AmbientOcclusion:
1884 fragmentShader.append(
" debugOutput += vec3(qt_aoFactor);\n");
1886 case QSSGRenderLayer::MaterialDebugMode::Normal:
1887 fragmentShader.append(
" debugOutput += qt_world_normal * 0.5 + 0.5;\n");
1889 case QSSGRenderLayer::MaterialDebugMode::Tangent:
1890 fragmentShader.append(
" debugOutput += qt_tangent * 0.5 + 0.5;\n");
1892 case QSSGRenderLayer::MaterialDebugMode::Binormal:
1893 fragmentShader.append(
" debugOutput += qt_binormal * 0.5 + 0.5;\n");
1895 case QSSGRenderLayer::MaterialDebugMode::F0:
1897 fragmentShader.append(
" debugOutput += qt_f0;");
1899 case QSSGRenderLayer::MaterialDebugMode::None:
1903 fragmentShader.append(
" fragOutput = vec4(debugOutput, 1.0);\n");
1906 if (shaderAugmentation.hasUserAugmentation())
1907 fragmentShader <<
" " << shaderAugmentation.body <<
";\n";
1991void QSSGMaterialShaderGenerator::setRhiMaterialProperties(
const QSSGRenderContextInterface &renderContext,
1992 QSSGRhiShaderPipeline &shaders,
1994 QSSGRhiGraphicsPipelineState *inPipelineState,
1995 const QSSGRenderGraphObject &inMaterial,
1996 const QSSGShaderDefaultMaterialKey &inKey,
1997 const QSSGShaderDefaultMaterialKeyProperties &inProperties,
1998 const QSSGRenderCameraList &inCameras,
1999 const QSSGRenderMvpArray &inModelViewProjections,
2000 const QMatrix3x3 &inNormalMatrix,
2001 const QMatrix4x4 &inGlobalTransform,
2002 const QMatrix4x4 &clipSpaceCorrMatrix,
2003 const QMatrix4x4 &localInstanceTransform,
2004 const QMatrix4x4 &globalInstanceTransform,
2005 const QSSGDataView<
float> &inMorphWeights,
2006 QSSGRenderableImage *inFirstImage,
2008 const QSSGLayerRenderData &inRenderProperties,
2009 const QSSGShaderLightListView &inLights,
2010 const QSSGShaderReflectionProbe &reflectionProbe,
2011 bool receivesShadows,
2012 bool receivesReflections,
2013 const QVector2D *shadowDepthAdjust,
2014 QRhiTexture *lightmapTexture)
2016 QSSGShaderMaterialAdapter *materialAdapter = getMaterialAdapter(inMaterial);
2017 QSSGRhiShaderPipeline::CommonUniformIndices &cui = shaders.commonUniformIndices;
2019 materialAdapter->setCustomPropertyUniforms(ubufData, shaders, renderContext);
2021 const QVector2D camProperties(inCameras[0]->clipPlanes);
2022 shaders.setUniform(ubufData,
"qt_cameraProperties", &camProperties, 2 *
sizeof(
float), &cui.cameraPropertiesIdx);
2024 const int viewCount = inCameras.count();
2027 QMatrix4x4 camGlobalTransforms[2] { QMatrix4x4{Qt::Uninitialized}, QMatrix4x4{Qt::Uninitialized} };
2028 if (viewCount < 2) {
2029 camGlobalTransforms[0] = inRenderProperties.getGlobalTransform(*inCameras[0]);
2031 for (size_t viewIndex = 0; viewIndex != std::size(camGlobalTransforms); ++viewIndex)
2032 camGlobalTransforms[viewIndex] = inRenderProperties.getGlobalTransform(*inCameras[viewIndex]);
2035 if (viewCount < 2) {
2036 const QMatrix4x4 &camGlobalTransform = camGlobalTransforms[0];
2037 const QVector3D camGlobalPos = QSSGRenderNode::getGlobalPos(camGlobalTransform);
2038 shaders.setUniform(ubufData,
"qt_cameraPosition", &camGlobalPos, 3 *
sizeof(
float), &cui.cameraPositionIdx);
2039 const QVector3D camDirection = QSSG_GUARD(inRenderProperties.renderedCameraData.has_value())
2040 ? inRenderProperties.renderedCameraData.value()[0].direction
2041 : QVector3D{ 0.0f, 0.0f, -1.0f };
2042 shaders.setUniform(ubufData,
"qt_cameraDirection", &camDirection, 3 *
sizeof(
float), &cui.cameraDirectionIdx);
2044 QVarLengthArray<QVector3D, 2> camGlobalPos(viewCount);
2045 QVarLengthArray<QVector3D> camDirection(viewCount);
2046 for (size_t viewIndex = 0; viewIndex != std::size(camGlobalTransforms); ++viewIndex) {
2047 const QMatrix4x4 &camGlobalTransform = camGlobalTransforms[viewIndex];
2048 camGlobalPos[viewIndex] = QSSGRenderNode::getGlobalPos(camGlobalTransform);
2049 camDirection[viewIndex] = QSSG_GUARD(inRenderProperties.renderedCameraData.has_value())
2050 ? inRenderProperties.renderedCameraData.value()[viewIndex].direction
2051 : QVector3D{ 0.0f, 0.0f, -1.0f };
2053 shaders.setUniformArray(ubufData,
"qt_cameraPosition", camGlobalPos.constData(), viewCount, QSSGRenderShaderValue::Vec3, &cui.cameraPositionIdx);
2054 shaders.setUniformArray(ubufData,
"qt_cameraDirection", camDirection.constData(), viewCount, QSSGRenderShaderValue::Vec3, &cui.cameraDirectionIdx);
2057 const auto globalRenderData = QSSGLayerRenderData::globalRenderProperties(renderContext);
2060 bool usesProjectionMatrix =
false;
2061 bool usesInvProjectionMatrix =
false;
2062 bool usesViewProjectionMatrix =
false;
2063 bool usesModelViewProjectionMatrix =
false;
2064 bool usesNormalMatrix =
false;
2065 bool usesParentMatrix =
false;
2067 if (inMaterial.type == QSSGRenderGraphObject::Type::CustomMaterial) {
2068 const auto *customMaterial =
static_cast<
const QSSGRenderCustomMaterial *>(&inMaterial);
2069 usesProjectionMatrix = customMaterial->m_renderFlags.testFlag(QSSGRenderCustomMaterial::RenderFlag::ProjectionMatrix);
2070 usesInvProjectionMatrix = customMaterial->m_renderFlags.testFlag(QSSGRenderCustomMaterial::RenderFlag::InverseProjectionMatrix);
2072 usesViewProjectionMatrix =
true;
2075 const bool usesInstancing = inProperties.m_usesInstancing.getValue(inKey);
2076 if (usesInstancing) {
2078 usesViewProjectionMatrix =
true;
2079 usesParentMatrix =
true;
2081 usesModelViewProjectionMatrix =
true;
2082 usesNormalMatrix =
true;
2085 if (materialAdapter->isTransmissionEnabled())
2086 usesViewProjectionMatrix =
true;
2089 if (usesProjectionMatrix || usesInvProjectionMatrix) {
2090 if (viewCount < 2) {
2091 const QMatrix4x4 projection = clipSpaceCorrMatrix * inCameras[0]->projection;
2092 if (usesProjectionMatrix)
2093 shaders.setUniform(ubufData,
"qt_projectionMatrix", projection.constData(), 16 *
sizeof(
float), &cui.projectionMatrixIdx);
2094 if (usesInvProjectionMatrix)
2095 shaders.setUniform(ubufData,
"qt_inverseProjectionMatrix", projection.inverted().constData(), 16 *
sizeof (
float), &cui.inverseProjectionMatrixIdx);
2097 QVarLengthArray<QMatrix4x4, 2> projections(viewCount);
2098 QVarLengthArray<QMatrix4x4, 2> invertedProjections(viewCount);
2099 for (size_t viewIndex = 0; viewIndex != std::size(camGlobalTransforms); ++viewIndex) {
2100 projections[viewIndex] = clipSpaceCorrMatrix * inCameras[viewIndex]->projection;
2101 if (usesInvProjectionMatrix)
2102 invertedProjections[viewIndex] = projections[viewIndex].inverted();
2104 if (usesProjectionMatrix)
2105 shaders.setUniformArray(ubufData,
"qt_projectionMatrix", projections.constData(), viewCount, QSSGRenderShaderValue::Matrix4x4, &cui.projectionMatrixIdx);
2106 if (usesInvProjectionMatrix)
2107 shaders.setUniformArray(ubufData,
"qt_inverseProjectionMatrix", invertedProjections.constData(), viewCount, QSSGRenderShaderValue::Matrix4x4, &cui.inverseProjectionMatrixIdx);
2111 if (viewCount < 2) {
2112 const QMatrix4x4 viewMatrix = camGlobalTransforms[0].inverted();
2113 shaders.setUniform(ubufData,
"qt_viewMatrix", viewMatrix.constData(), 16 *
sizeof(
float), &cui.viewMatrixIdx);
2115 QVarLengthArray<QMatrix4x4, 2> viewMatrices(viewCount);
2116 for (size_t viewIndex = 0; viewIndex != std::size(camGlobalTransforms); ++viewIndex)
2117 viewMatrices[viewIndex] = camGlobalTransforms[viewIndex].inverted();
2118 shaders.setUniformArray(ubufData,
"qt_viewMatrix", viewMatrices.constData(), viewCount, QSSGRenderShaderValue::Matrix4x4, &cui.viewMatrixIdx);
2121 if (usesViewProjectionMatrix) {
2122 if (viewCount < 2) {
2123 const QMatrix4x4 &camGlobalTransform = camGlobalTransforms[0];
2124 QMatrix4x4 viewProj(Qt::Uninitialized);
2125 inCameras[0]->calculateViewProjectionMatrix(camGlobalTransform, viewProj);
2126 viewProj = clipSpaceCorrMatrix * viewProj;
2127 shaders.setUniform(ubufData,
"qt_viewProjectionMatrix", viewProj.constData(), 16 *
sizeof(
float), &cui.viewProjectionMatrixIdx);
2129 QVarLengthArray<QMatrix4x4, 2> viewProjections(viewCount);
2130 for (size_t viewIndex = 0; viewIndex != std::size(camGlobalTransforms); ++viewIndex) {
2131 const auto &camGlobalTransform = camGlobalTransforms[viewIndex];
2132 inCameras[viewIndex]->calculateViewProjectionMatrix(camGlobalTransform, viewProjections[viewIndex]);
2133 viewProjections[viewIndex] = clipSpaceCorrMatrix * viewProjections[viewIndex];
2135 shaders.setUniformArray(ubufData,
"qt_viewProjectionMatrix", viewProjections.constData(), viewCount, QSSGRenderShaderValue::Matrix4x4, &cui.viewProjectionMatrixIdx);
2141 shaders.setUniform(ubufData,
"qt_modelMatrix", localInstanceTransform.constData(), 16 *
sizeof(
float), &cui.modelMatrixIdx);
2143 shaders.setUniform(ubufData,
"qt_modelMatrix", inGlobalTransform.constData(), 16 *
sizeof(
float), &cui.modelMatrixIdx);
2145 if (usesModelViewProjectionMatrix) {
2146 if (viewCount < 2) {
2147 QMatrix4x4 mvp { clipSpaceCorrMatrix };
2148 mvp *= inModelViewProjections[0];
2149 shaders.setUniform(ubufData,
"qt_modelViewProjection", mvp.constData(), 16 *
sizeof(
float), &cui.modelViewProjectionIdx);
2151 QVarLengthArray<QMatrix4x4, 2> mvps(viewCount);
2152 for (
int viewIndex = 0; viewIndex < viewCount; ++viewIndex)
2153 mvps[viewIndex] = clipSpaceCorrMatrix * inModelViewProjections[viewIndex];
2154 shaders.setUniformArray(ubufData,
"qt_modelViewProjection", mvps.constData(), viewCount, QSSGRenderShaderValue::Matrix4x4, &cui.modelViewProjectionIdx);
2157 if (usesNormalMatrix)
2158 shaders.setUniform(ubufData,
"qt_normalMatrix", inNormalMatrix.constData(), 12 *
sizeof(
float), &cui.normalMatrixIdx,
2159 QSSGRhiShaderPipeline::UniformFlag::Mat3);
2160 if (usesParentMatrix)
2161 shaders.setUniform(ubufData,
"qt_parentMatrix", globalInstanceTransform.constData(), 16 *
sizeof(
float));
2164 const qsizetype morphSize = inProperties.m_targetCount.getValue(inKey);
2165 if (morphSize > 0) {
2166 if (inMorphWeights.mSize >= morphSize) {
2167 shaders.setUniformArray(ubufData,
"qt_morphWeights", inMorphWeights.mData, morphSize,
2168 QSSGRenderShaderValue::Float, &cui.morphWeightsIdx);
2170 const QList<
float> zeroWeights(morphSize - inMorphWeights.mSize, 0.0f);
2171 QList<
float> newWeights(inMorphWeights.mData, inMorphWeights.mData + inMorphWeights.mSize);
2172 newWeights.append(zeroWeights);
2173 shaders.setUniformArray(ubufData,
"qt_morphWeights", newWeights.constData(), morphSize,
2174 QSSGRenderShaderValue::Float, &cui.morphWeightsIdx);
2178 QVector3D theLightAmbientTotal;
2179 quint32 lightCount = 0;
2180 quint32 directionalLightCount = 0;
2181 quint32 shadowCount = 0;
2182 quint32 directionalShadowCount = 0;
2183 QSSGShaderLightsUniformData &lightsUniformData(shaders.lightsUniformData());
2184 QSSGShaderDirectionalLightsUniformData &directionalLightsUniformData(shaders.directionalLightsUniformData());
2186 for (quint32 lightIdx = 0, lightEnd = inLights.size();
2187 lightIdx < lightEnd && lightIdx < QSSG_MAX_NUM_LIGHTS; ++lightIdx)
2189 QSSGRenderLight *theLight(inLights[lightIdx].light);
2192 const bool lightShadows = inLights[lightIdx].shadows;
2193 const float brightness = theLight->m_brightness;
2194 quint32 lightmapState = 0;
2195 if (theLight->m_bakingEnabled) {
2196 if (theLight->m_fullyBaked) {
2205 const QVector3D diffuseColor(theLight->m_diffuseColor.x() * brightness,
2206 theLight->m_diffuseColor.y() * brightness,
2207 theLight->m_diffuseColor.z() * brightness);
2208 const QVector3D specularColor(theLight->m_specularColor.x() * brightness,
2209 theLight->m_specularColor.y() * brightness,
2210 theLight->m_specularColor.z() * brightness);
2211 const QVector3D direction(inLights[lightIdx].direction);
2214 if (theLight->type == QSSGRenderGraphObject::Type::DirectionalLight) {
2216 QSSGShaderDirectionalLightData &lightData(directionalLightsUniformData.directionalLightData[directionalLightCount]);
2217 lightData.direction[0] = direction.x();
2218 lightData.direction[1] = direction.y();
2219 lightData.direction[2] = direction.z();
2220 lightData.diffuseColor[0] = diffuseColor.x();
2221 lightData.diffuseColor[1] = diffuseColor.y();
2222 lightData.diffuseColor[2] = diffuseColor.z();
2223 lightData.specularColor[0] = specularColor.x();
2224 lightData.specularColor[1] = specularColor.y();
2225 lightData.specularColor[2] = specularColor.z();
2226 lightData.lightmapState = lightmapState;
2227 if (lightShadows && receivesShadows) {
2228 lightData.enableShadows = 1.0f;
2229 QSSGShadowMapEntry *pEntry = inRenderProperties.getShadowMapManager()->shadowMapEntry(lightIdx);
2232 const quint32 layerCount = pEntry->m_csmNumSplits + 1;
2234 for (quint32 i = 0; i < layerCount; ++i)
2235 memcpy(lightData.matrices[i], pEntry->m_fixedScaleBiasMatrix[i].constData(), 16 *
sizeof(
float));
2237 lightData.shadowBias = theLight->m_shadowBias;
2239 const bool noCascades = !(pEntry->m_csmActive[0] || pEntry->m_csmActive[1] || pEntry->m_csmActive[2] || pEntry->m_csmActive[3]);
2240 if (theLight->type == QSSGRenderLight::Type::DirectionalLight && noCascades)
2241 lightData.enableShadows = 0.0f;
2242 lightData.shadowFactor = theLight->m_shadowFactor;
2243 lightData.shadowMapFar = theLight->m_shadowMapFar;
2244 lightData.shadowPcfSamples = softShadowQualityToInt(theLight->m_softShadowQuality);
2245 lightData.shadowPcfFactor = theLight->m_pcfFactor;
2247 for (quint32 i = 0; i < layerCount; ++i) {
2248 const auto &atlasInfo = pEntry->m_atlasInfo[i];
2249 lightData.atlasLocations[i][0] = atlasInfo.uOffset;
2250 lightData.atlasLocations[i][1] = atlasInfo.vOffset;
2251 lightData.atlasLocations[i][2] = atlasInfo.uvScale;
2252 lightData.atlasLocations[i][3] = atlasInfo.layerIndex;
2255 lightData.csmNumSplits = pEntry->m_csmNumSplits;
2256 memcpy(lightData.csmSplits, pEntry->m_csmSplits, 4 *
sizeof(
float));
2257 memcpy(lightData.csmActive, pEntry->m_csmActive, 4 *
sizeof(
float));
2258 lightData.csmBlendRatio = theLight->m_csmBlendRatio;
2259 for (quint32 i = 0; i < layerCount; ++i)
2260 memcpy(lightData.dimensionsInverted[i], &pEntry->m_dimensionsInverted[i], 4 *
sizeof(
float));
2262 directionalShadowCount++;
2264 lightData.enableShadows = 0.0f;
2266 directionalLightCount++;
2269 QSSGShaderLightData &lightData(lightsUniformData.lightData[lightCount]);
2270 const auto gt = inRenderProperties.getGlobalTransform(*theLight);
2271 const QVector3D globalPos = QSSGRenderNode::getGlobalPos(gt);
2272 lightData.position[0] = globalPos.x();
2273 lightData.position[1] = globalPos.y();
2274 lightData.position[2] = globalPos.z();
2275 lightData.constantAttenuation = QSSGUtils::aux::translateConstantAttenuation(theLight->m_constantFade);
2276 lightData.linearAttenuation = QSSGUtils::aux::translateLinearAttenuation(theLight->m_linearFade);
2277 lightData.quadraticAttenuation = QSSGUtils::aux::translateQuadraticAttenuation(theLight->m_quadraticFade);
2278 lightData.coneAngle = 360.0f;
2279 lightData.direction[0] = direction.x();
2280 lightData.direction[1] = direction.y();
2281 lightData.direction[2] = direction.z();
2282 lightData.diffuseColor[0] = diffuseColor.x();
2283 lightData.diffuseColor[1] = diffuseColor.y();
2284 lightData.diffuseColor[2] = diffuseColor.z();
2285 lightData.specularColor[0] = specularColor.x();
2286 lightData.specularColor[1] = specularColor.y();
2287 lightData.specularColor[2] = specularColor.z();
2288 lightData.lightmapState = lightmapState;
2289 if (theLight->type == QSSGRenderLight::Type::SpotLight) {
2293 const float coneAngle = theLight->m_coneAngle;
2294 const float innerConeAngle = (theLight->m_innerConeAngle > coneAngle) ? coneAngle : theLight->m_innerConeAngle;
2295 lightData.coneAngle = qCos(qDegreesToRadians(coneAngle));
2296 lightData.innerConeAngle = qCos(qDegreesToRadians(innerConeAngle));
2299 if (lightShadows && receivesShadows) {
2300 QSSGShadowMapEntry *pEntry = inRenderProperties.getShadowMapManager()->shadowMapEntry(lightIdx);
2302 lightData.enableShadows = 1.0f;
2303 lightData.shadowPcfFactor = theLight->m_pcfFactor;
2304 lightData.shadowPcfSamples = softShadowQualityToInt(theLight->m_softShadowQuality);
2305 lightData.shadowFactor = theLight->m_shadowFactor;
2306 lightData.shadowBias = theLight->m_shadowBias;
2307 lightData.shadowClipNear = 1.0f;
2308 lightData.shadowMapFar = pEntry->m_shadowMapFar;
2309 lightData.shadowTextureSize = pEntry->m_atlasInfo[0].uvScale;
2311 if (theLight->type == QSSGRenderLight::Type::SpotLight) {
2313 static const QMatrix4x4 bias = {
2317 0.0, 0.0, 0.0, 1.0 };
2318 const QMatrix4x4 m = bias * pEntry->m_lightViewProjection[0];
2319 memcpy(lightData.shadowMatrix, m.constData(), 16 *
sizeof(
float));
2320 lightData.shadowAtlasUV0[0] = pEntry->m_atlasInfo[0].uOffset;
2321 lightData.shadowAtlasUV0[1] = pEntry->m_atlasInfo[0].vOffset;
2322 lightData.shadowAtlasLayer0 = pEntry->m_atlasInfo[0].layerIndex;
2325 Q_ASSERT(theLight->type == QSSGRenderLight::Type::PointLight);
2326 memcpy(lightData.shadowMatrix, pEntry->m_lightView.constData(), 16 *
sizeof(
float));
2327 lightData.shadowAtlasUV0[0] = pEntry->m_atlasInfo[0].uOffset;
2328 lightData.shadowAtlasUV0[1] = pEntry->m_atlasInfo[0].vOffset;
2329 lightData.shadowAtlasLayer0 = pEntry->m_atlasInfo[0].layerIndex;
2330 lightData.shadowAtlasUV1[0] = pEntry->m_atlasInfo[1].uOffset;
2331 lightData.shadowAtlasUV1[1] = pEntry->m_atlasInfo[1].vOffset;
2332 lightData.shadowAtlasLayer1 = pEntry->m_atlasInfo[1].layerIndex;
2337 lightData.enableShadows = 0.0f;
2343 theLightAmbientTotal += theLight->m_ambientColor;
2347 if (shadowCount > 0 || directionalShadowCount > 0) {
2348 shaders.setShadowMapAtlasTexture(inRenderProperties.getShadowMapManager()->shadowMapAtlasTexture());
2349 shaders.setShadowMapBlueNoiseTexture(inRenderProperties.getShadowMapManager()->shadowMapBlueNoiseTexture());
2351 shaders.setShadowMapAtlasTexture(
nullptr);
2352 shaders.setShadowMapBlueNoiseTexture(
nullptr);
2355 const QSSGRhiRenderableTexture *depthTexture = inRenderProperties.getRenderResult(QSSGRenderResult::Key::DepthTexture);
2356 const QSSGRhiRenderableTexture *normalTexture = inRenderProperties.getRenderResult(QSSGRenderResult::Key::NormalTexture);
2357 const QSSGRhiRenderableTexture *ssaoTexture = inRenderProperties.getRenderResult(QSSGRenderResult::Key::AoTexture);
2358 const QSSGRhiRenderableTexture *screenTexture = inRenderProperties.getRenderResult(QSSGRenderResult::Key::ScreenTexture);
2359 const QSSGRhiRenderableTexture *motionVectorTexture = inRenderProperties.getRenderResult(QSSGRenderResult::Key::MotionVectorTexture);
2361 shaders.setDepthTexture(depthTexture->texture);
2362 shaders.setNormalTexture(normalTexture->texture);
2363 shaders.setSsaoTexture(ssaoTexture->texture);
2364 shaders.setScreenTexture(screenTexture->texture);
2365 shaders.setLightmapTexture(lightmapTexture);
2366 shaders.setMotionVectorTexture(motionVectorTexture->texture);
2368#ifdef QSSG_OIT_USE_BUFFERS
2369 shaders.setOITImages((QRhiTexture*)inRenderProperties.getOitRenderContextConst().aBuffer,
2370 (QRhiTexture*)inRenderProperties.getOitRenderContextConst().auxBuffer,
2371 (QRhiTexture*)inRenderProperties.getOitRenderContextConst().counterBuffer);
2372 if (inRenderProperties.getOitRenderContextConst().aBuffer) {
2374 const QSSGRhiRenderableTexture *abuf = inRenderProperties.getRenderResult(QSSGRenderResult::Key::ABufferImage);
2375 const QSSGRhiRenderableTexture *aux = inRenderProperties.getRenderResult(QSSGRenderResult::Key::AuxiliaryImage);
2376 const QSSGRhiRenderableTexture *counter = inRenderProperties.getRenderResult(QSSGRenderResult::Key::CounterImage);
2377 shaders.setOITImages(abuf->texture, aux->texture, counter->texture);
2378 if (abuf->texture) {
2380 int abufWidth = RenderHelpers::rhiCalculateABufferSize(inRenderProperties.layer.oitNodeCount);
2381 int listNodeCount = abufWidth * abufWidth;
2382 shaders.setUniform(ubufData,
"qt_ABufImageWidth", &abufWidth,
sizeof(
int), &cui.abufImageWidth);
2383 shaders.setUniform(ubufData,
"qt_listNodeCount", &listNodeCount,
sizeof(
int), &cui.listNodeCount);
2384 int viewSize[2] = {inRenderProperties.layerPrepResult.textureDimensions().width(), inRenderProperties.layerPrepResult.textureDimensions().height()};
2385 shaders.setUniform(ubufData,
"qt_viewSize", viewSize,
sizeof(
int) * 2, &cui.viewSize);
2386 int samples = inPipelineState->samples;
2387 shaders.setUniform(ubufData,
"qt_samples", &samples,
sizeof(
int), &cui.samples);
2390 QSSGRenderImageTexture resolvedLayerIbl;
2391 QSSGRenderTextureCoordOp hTile = QSSGRenderTextureCoordOp::ClampToEdge;
2392 QSSGRenderTextureCoordOp vTile = QSSGRenderTextureCoordOp::ClampToEdge;
2393 const QSSGRenderLayer &layer = QSSGLayerRenderData::getCurrent(*renderContext.renderer())->layer;
2394 if (layer.lightProbe) {
2395 resolvedLayerIbl = renderContext.bufferManager()->loadRenderImage(layer.lightProbe, QSSGBufferManager::MipModeBsdf);
2396 hTile = layer.lightProbe->m_horizontalTilingMode;
2397 vTile = layer.lightProbe->m_verticalTilingMode;
2398 }
else if (layer.skyMaterial && layer.skyMaterial->enableIBL) {
2399 resolvedLayerIbl = inRenderProperties.skyMaterialTexture;
2402 const auto &lightProbeData = layer.lightProbeSettings;
2405 QSSGRenderImage *materialIblProbe = materialAdapter->iblProbe();
2407 QSSGRenderImageTexture lightProbeTexture;
2408 if (materialIblProbe) {
2409 lightProbeTexture = renderContext.bufferManager()->loadRenderImage(materialIblProbe, QSSGBufferManager::MipModeBsdf);
2410 hTile = materialIblProbe->m_horizontalTilingMode;
2411 vTile = materialIblProbe->m_verticalTilingMode;
2412 }
else if (resolvedLayerIbl.m_texture) {
2413 lightProbeTexture = resolvedLayerIbl;
2416 if (lightProbeTexture.m_texture) {
2417 const int maxMipLevel = lightProbeTexture.m_mipmapCount - 1;
2419 if (!materialIblProbe && !lightProbeData.probeOrientation.isIdentity()) {
2420 shaders.setUniform(ubufData,
"qt_lightProbeOrientation",
2421 lightProbeData.probeOrientation.constData(),
2422 12 *
sizeof(
float), &cui.lightProbeOrientationIdx,
2423 QSSGRhiShaderPipeline::UniformFlag::Mat3);
2426 const float props[4] = { 0.0f,
float(maxMipLevel), lightProbeData.probeHorizon, lightProbeData.probeExposure };
2427 shaders.setUniform(ubufData,
"qt_lightProbeProperties", props, 4 *
sizeof(
float), &cui.lightProbePropertiesIdx);
2429 shaders.setLightProbeTexture(lightProbeTexture.m_texture, hTile, vTile);
2432 const float emptyProps[4] = { 0.0f, 0.0f, -1.0f, 0.0f };
2433 shaders.setUniform(ubufData,
"qt_lightProbeProperties", emptyProps, 4 *
sizeof(
float), &cui.lightProbePropertiesIdx);
2435 shaders.setLightProbeTexture(
nullptr);
2438 if (receivesReflections && reflectionProbe.enabled) {
2439 shaders.setUniform(ubufData,
"qt_reflectionProbeCubeMapCenter", &reflectionProbe.probeCubeMapCenter, 3 *
sizeof(
float), &cui.reflectionProbeCubeMapCenter);
2440 shaders.setUniform(ubufData,
"qt_reflectionProbeBoxMin", &reflectionProbe.probeBoxMin, 3 *
sizeof(
float), &cui.reflectionProbeBoxMin);
2441 shaders.setUniform(ubufData,
"qt_reflectionProbeBoxMax", &reflectionProbe.probeBoxMax, 3 *
sizeof(
float), &cui.reflectionProbeBoxMax);
2442 shaders.setUniform(ubufData,
"qt_reflectionProbeCorrection", &reflectionProbe.parallaxCorrection,
sizeof(
int), &cui.reflectionProbeCorrection);
2445 const QVector3D emissiveColor = materialAdapter->emissiveColor();
2446 shaders.setUniform(ubufData,
"qt_material_emissive_color", &emissiveColor, 3 *
sizeof(
float), &cui.material_emissiveColorIdx);
2448 const auto qMix = [](
float x,
float y,
float a) {
2449 return (x * (1.0f - a) + (y * a));
2452 const auto qMix3 = [&qMix](
const QVector3D &x,
const QVector3D &y,
float a) {
2453 return QVector3D{qMix(x.x(), y.x(), a), qMix(x.y(), y.y(), a), qMix(x.z(), y.z(), a)};
2456 const QVector4D color = materialAdapter->color();
2457 const QVector3D materialSpecularTint = materialAdapter->specularTint();
2458 const QVector3D specularTint = materialAdapter->isPrincipled() ? qMix3(QVector3D(1.0f, 1.0f, 1.0f), color.toVector3D(), materialSpecularTint.x())
2459 : materialSpecularTint;
2460 shaders.setUniform(ubufData,
"qt_material_base_color", &color, 4 *
sizeof(
float), &cui.material_baseColorIdx);
2462 const float ior = materialAdapter->ior();
2463 QVector4D specularColor(specularTint, ior);
2464 shaders.setUniform(ubufData,
"qt_material_specular", &specularColor, 4 *
sizeof(
float), &cui.material_specularIdx);
2466 const bool hasLighting = materialAdapter->hasLighting();
2467 shaders.setLightsEnabled(hasLighting);
2469 const float lightAndShadowCounts[4] = {
2471 float(directionalLightCount),
2473 float(directionalShadowCount)
2475 shaders.setUniform(ubufData,
"qt_lightAndShadowCounts", &lightAndShadowCounts, 4 *
sizeof(
float), &cui.lightAndShadowCountsIdx);
2477 const size_t lightDataSize = lightCount *
sizeof(QSSGShaderLightData);
2478 const size_t directionalLightDataSize = directionalLightCount *
sizeof(QSSGShaderDirectionalLightData);
2480 memcpy(ubufData + shaders.ub0LightDataOffset(), &lightsUniformData, lightDataSize);
2481 memcpy(ubufData + shaders.ub0DirectionalLightDataOffset(), &directionalLightsUniformData, directionalLightDataSize);
2484 shaders.setUniform(ubufData,
"qt_light_ambient_total", &theLightAmbientTotal, 3 *
sizeof(
float), &cui.light_ambient_totalIdx);
2486 const float materialProperties[4] = {
2487 materialAdapter->specularAmount(),
2488 materialAdapter->specularRoughness(),
2489 materialAdapter->metalnessAmount(),
2492 shaders.setUniform(ubufData,
"qt_material_properties", materialProperties, 4 *
sizeof(
float), &cui.material_propertiesIdx);
2494 const float materialProperties2[4] = {
2495 materialAdapter->fresnelPower(),
2496 materialAdapter->bumpAmount(),
2497 materialAdapter->translucentFallOff(),
2498 materialAdapter->diffuseLightWrap()
2500 shaders.setUniform(ubufData,
"qt_material_properties2", materialProperties2, 4 *
sizeof(
float), &cui.material_properties2Idx);
2502 const float materialProperties3[4] = {
2503 materialAdapter->occlusionAmount(),
2504 materialAdapter->alphaCutOff(),
2505 materialAdapter->clearcoatAmount(),
2506 materialAdapter->clearcoatRoughnessAmount()
2508 shaders.setUniform(ubufData,
"qt_material_properties3", materialProperties3, 4 *
sizeof(
float), &cui.material_properties3Idx);
2510 const float materialProperties4[4] = {
2511 materialAdapter->heightAmount(),
2512 materialAdapter->minHeightSamples(),
2513 materialAdapter->maxHeightSamples(),
2514 materialAdapter->transmissionFactor()
2516 shaders.setUniform(ubufData,
"qt_material_properties4", materialProperties4, 4 *
sizeof(
float), &cui.material_properties4Idx);
2518 const bool hasCustomFrag = materialAdapter->hasCustomShaderSnippet(QSSGShaderCache::ShaderType::Fragment);
2519 if (!hasCustomFrag) {
2520 if (inProperties.m_fresnelScaleBiasEnabled.getValue(inKey) || inProperties.m_clearcoatFresnelScaleBiasEnabled.getValue(inKey)) {
2521 const float materialProperties5[4] = {
2522 materialAdapter->fresnelScale(),
2523 materialAdapter->fresnelBias(),
2524 materialAdapter->clearcoatFresnelScale(),
2525 materialAdapter->clearcoatFresnelBias()
2527 shaders.setUniform(ubufData,
"qt_material_properties5", materialProperties5, 4 *
sizeof(
float), &cui.material_properties5Idx);
2530 const float material_clearcoat_normal_strength = materialAdapter->clearcoatNormalStrength();
2531 shaders.setUniform(ubufData,
"qt_material_clearcoat_normal_strength", &material_clearcoat_normal_strength,
sizeof(
float), &cui.clearcoatNormalStrengthIdx);
2533 const float material_clearcoat_fresnel_power = materialAdapter->clearcoatFresnelPower();
2534 shaders.setUniform(ubufData,
"qt_material_clearcoat_fresnel_power", &material_clearcoat_fresnel_power,
sizeof(
float), &cui.clearcoatFresnelPowerIdx);
2536 if (materialAdapter->isTransmissionEnabled()) {
2537 const QVector4D attenuationProperties(materialAdapter->attenuationColor(), materialAdapter->attenuationDistance());
2538 shaders.setUniform(ubufData,
"qt_material_attenuation", &attenuationProperties, 4 *
sizeof(
float), &cui.material_attenuationIdx);
2540 const float thickness = materialAdapter->thicknessFactor();
2541 shaders.setUniform(ubufData,
"qt_material_thickness", &thickness,
sizeof(
float), &cui.thicknessFactorIdx);
2545 const float rhiProperties[4] = {
2546 globalRenderData.isYUpInFramebuffer ? 1.0f : -1.0f,
2547 globalRenderData.isYUpInNDC ? 1.0f : -1.0f,
2548 globalRenderData.isClipDepthZeroToOne ? 0.0f : -1.0f,
2551 shaders.setUniform(ubufData,
"qt_rhi_properties", rhiProperties, 4 *
sizeof(
float), &cui.rhiPropertiesIdx);
2553 qsizetype imageIdx = 0;
2554 for (QSSGRenderableImage *theImage = inFirstImage; theImage; theImage = theImage->m_nextImage, ++imageIdx) {
2556 const auto &names = imageStringTable[
int(theImage->m_mapType)];
2557 if (imageIdx == cui.imageIndices.size())
2558 cui.imageIndices.append(QSSGRhiShaderPipeline::CommonUniformIndices::ImageIndices());
2559 auto &indices = cui.imageIndices[imageIdx];
2561 const QMatrix4x4 &textureTransform = theImage->m_imageNode.m_textureTransform;
2564 const float *dataPtr(textureTransform.constData());
2568 const float offsets[3] = { dataPtr[12], dataPtr[13], 0.0f };
2569 shaders.setUniform(ubufData, names.imageOffsets, offsets,
sizeof(offsets), &indices.imageOffsetsUniformIndex);
2571 const float rotations[4] = { dataPtr[0], dataPtr[4], dataPtr[1], dataPtr[5] };
2572 shaders.setUniform(ubufData, names.imageRotations, rotations,
sizeof(rotations), &indices.imageRotationsUniformIndex);
2575 if (shadowDepthAdjust)
2576 shaders.setUniform(ubufData,
"qt_shadowDepthAdjust", shadowDepthAdjust, 2 *
sizeof(
float), &cui.shadowDepthAdjustIdx);
2578 const bool usesPointsTopology = inProperties.m_usesPointsTopology.getValue(inKey);
2579 if (usesPointsTopology) {
2580 const float pointSize = materialAdapter->pointSize();
2581 shaders.setUniform(ubufData,
"qt_materialPointSize", &pointSize,
sizeof(
float), &cui.pointSizeIdx);
2588 if (inRenderProperties.layer.fog.enabled) {
2589 const float fogColor[4] = {
2590 inRenderProperties.layer.fog.color.x(),
2591 inRenderProperties.layer.fog.color.y(),
2592 inRenderProperties.layer.fog.color.z(),
2593 inRenderProperties.layer.fog.density
2595 shaders.setUniform(ubufData,
"qt_fogColor", fogColor, 4 *
sizeof(
float), &cui.fogColorIdx);
2596 const float fogDepthProperties[4] = {
2597 inRenderProperties.layer.fog.depthBegin,
2598 inRenderProperties.layer.fog.depthEnd,
2599 inRenderProperties.layer.fog.depthCurve,
2600 inRenderProperties.layer.fog.depthEnabled ? 1.0f : 0.0f
2602 shaders.setUniform(ubufData,
"qt_fogDepthProperties", fogDepthProperties, 4 *
sizeof(
float), &cui.fogDepthPropertiesIdx);
2603 const float fogHeightProperties[4] = {
2604 inRenderProperties.layer.fog.heightMin,
2605 inRenderProperties.layer.fog.heightMax,
2606 inRenderProperties.layer.fog.heightCurve,
2607 inRenderProperties.layer.fog.heightEnabled ? 1.0f : 0.0f
2609 shaders.setUniform(ubufData,
"qt_fogHeightProperties", fogHeightProperties, 4 *
sizeof(
float), &cui.fogHeightPropertiesIdx);
2610 const float fogTransmitProperties[4] = {
2611 inRenderProperties.layer.fog.transmitCurve,
2614 inRenderProperties.layer.fog.transmitEnabled ? 1.0f : 0.0f
2616 shaders.setUniform(ubufData,
"qt_fogTransmitProperties", fogTransmitProperties, 4 *
sizeof(
float), &cui.fogTransmitPropertiesIdx);
2619 inPipelineState->lineWidth = materialAdapter->lineWidth();