mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 19:26:39 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			121 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
| void main_vp
 | |
| (
 | |
|     in float4 iPos : POSITION
 | |
|     , in float2 iUv : TEXCOORD0
 | |
| 
 | |
|     , out float4 oPos : POSITION
 | |
|     , out float3 oScreenCoords : TEXCOORD0
 | |
|     , out float2 oUv : TEXCOORD1
 | |
|     , out float oDepth : TEXCOORD2
 | |
|     , out float4 oEyeVector : TEXCOORD3
 | |
| 
 | |
|     , uniform float4x4 wvpMat
 | |
|     , uniform float4 camPosObjSpace
 | |
| )
 | |
| {
 | |
|     oPos = mul(wvpMat, iPos);
 | |
| 
 | |
|     oUv = iUv * 10; // uv scale
 | |
|     oDepth = oPos.z;
 | |
| 
 | |
|     float4x4 scalemat = float4x4(   0.5,    0,      0,      0.5,
 | |
|                                     0,      -0.5,   0,      0.5,
 | |
|                                     0,      0,      0.5,    0.5,
 | |
|                                     0,      0,      0,      1   );
 | |
|     float4 texcoordProj = mul(scalemat, oPos);
 | |
|     oScreenCoords = float3(texcoordProj.x, texcoordProj.y, texcoordProj.w);
 | |
| 
 | |
|     oEyeVector = camPosObjSpace - iPos;
 | |
| }
 | |
| 
 | |
| void main_fp
 | |
| (
 | |
|     out float4 oColor : COLOR
 | |
| 
 | |
|     , in float3 iScreenCoords : TEXCOORD0
 | |
|     , in float2 iUv : TEXCOORD1
 | |
|     , in float iDepth : TEXCOORD2
 | |
|     , in float4 iEyeVector : TEXCOORD3
 | |
|     , uniform float renderTargetFlipping
 | |
|     , uniform float4 lightPosObjSpace0
 | |
|     , uniform float4 lightSpecularColour0
 | |
| 
 | |
|     , uniform sampler2D reflectionMap : register(s0)
 | |
|     , uniform sampler2D refractionMap : register(s1)
 | |
|     , uniform sampler2D depthMap : register(s2)
 | |
|     , uniform sampler2D normalMap : register(s3)
 | |
|     , uniform float time
 | |
|     , uniform float far
 | |
|     , uniform float4 fogParams
 | |
|     , uniform float4 fogColour
 | |
|     , uniform float isUnderwater
 | |
| )
 | |
| {
 | |
| 
 | |
|     float2 screenCoords = iScreenCoords.xy / iScreenCoords.z;
 | |
|     screenCoords.y = (1-saturate(renderTargetFlipping))+renderTargetFlipping*screenCoords.y;
 | |
| 
 | |
|     // No need for transparency since we are using a refraction map
 | |
|     oColor.a = 1;
 | |
| 
 | |
|     // Sample screen-space depth map and subtract pixel depth to get the real water depth
 | |
|     float depthTex = tex2D(depthMap, screenCoords).r;
 | |
|     float depth1 = depthTex * far - iDepth;
 | |
|     depth1 = saturate(depth1 / 500.f);
 | |
| 
 | |
|     // Simple wave effect. to be replaced by something better
 | |
|     float2 uv1 = iUv + time * float2(0.5, 0);
 | |
|     float2 uv2 = iUv + time * float2(0, 0.5);
 | |
|     float2 uv3 = iUv + time * float2(-0.5, 0);
 | |
|     float2 uv4 = iUv + time * float2(0, -0.5);
 | |
|     float4 normal = tex2D(normalMap, uv1) + tex2D(normalMap, uv2) + tex2D(normalMap, uv3) + tex2D(normalMap, uv4);
 | |
|     normal = normal / 4.f;
 | |
|     normal = 2*normal - 1;
 | |
| 
 | |
|     float2 screenCoords_reflect = screenCoords + normal.yx * 0.05;
 | |
|     float2 screenCoords_refract = screenCoords + normal.yx * 0.05 * depth1;
 | |
| 
 | |
|     // Sample depth again with the refracted coordinates
 | |
|     depthTex = tex2D(depthMap, screenCoords_refract).r;
 | |
|     float depth2 = (depthTex * far - iDepth) / 500.f;
 | |
|     depth2 = (depthTex == 0 ? 1 : depth2);
 | |
|     // if depth2 is less than 0, this means we would refract something which is above water,
 | |
|     // which we don't want to - so in that case, don't refract
 | |
|     if (depth2 < 0.25) // delta due to inaccuracies
 | |
|     {
 | |
|         screenCoords_refract = screenCoords;
 | |
|         depth2 = depth1;
 | |
|     }
 | |
|     depth2 = saturate(depth2);
 | |
| 
 | |
|     float4 reflection = tex2D(reflectionMap, screenCoords_reflect);
 | |
|     float4 refraction = tex2D(refractionMap, screenCoords_refract);
 | |
| 
 | |
|     // tangent to object space
 | |
|     normal.xyz = normal.xzy;
 | |
| 
 | |
|     iEyeVector.xyz = normalize(iEyeVector.xyz);
 | |
| 
 | |
|     // fresnel
 | |
|     float facing = 1.0 - max(abs(dot(iEyeVector.xyz, normal.xyz)), 0);
 | |
|     float reflectionFactor = saturate(0.35 + 0.65 * pow(facing, 2));
 | |
| 
 | |
|     // specular
 | |
|     float3 lightDir = normalize(lightPosObjSpace0.xyz); // assumes that light 0 is a directional light
 | |
|     float3 halfVector = normalize(iEyeVector + lightDir);
 | |
|     float specular = pow(max(dot(normal.xyz, halfVector.xyz), 0), 64);
 | |
| 
 | |
|     float opacity = depth2 * saturate(reflectionFactor + specular);
 | |
|     opacity *= (1-isUnderwater);
 | |
| 
 | |
|     reflection.xyz += lightSpecularColour0.xyz * specular;
 | |
| 
 | |
|     oColor.xyz = lerp(refraction.xyz, reflection.xyz, opacity);
 | |
| 
 | |
|     oColor.xyz += isUnderwater * float3(0, 0.35, 0.35); // underwater tint color
 | |
|     oColor.xyz = lerp(oColor.xyz, float3(0, 0.65, 0.65), saturate(isUnderwater * (iDepth / 2000.f))); // underwater fog
 | |
| 
 | |
|     // add fog
 | |
|     //float fogValue = saturate((iDepth - fogParams.y) * fogParams.w);
 | |
|     //oColor.xyz = lerp(oColor.xyz, fogColour, fogValue);
 | |
| }
 |