UnityShader案例(五)——法线纹理
来源:互联网 发布:高维数据分析 编辑:程序博客网 时间:2024/05/21 21:38
一、在切线空间下,法线纹理映射的shader实现
1、基本思想
在片元着色器中通过对纹理的采样得到切线空间下的法线,然后再与切线空间下的视角方向,光照方向等进行计算,得到最终的光照结果。
2、源码
Shader "Custom/Texture/TangentNormalTextute"{ Properties { //主纹理的贴图 _MainTex ("Texture", 2D) = "white" {} _Color("Color",Color)=(1,1,1,1) //法线贴图 _BumpMap("NormalMap",2D)="white"{} //控制凹凸程度 _BumpScale("BumpScale",Float)=1.0 //高光颜色 _Specular("Specular",Color)=(1,1,1,1) //高光系数 _Gloss("Gloss",Range(8.0,256))=20 } SubShader { Tags { "RenderType"="Opaque" } LOD 100 Pass { Tags{"LightMode"="ForwardBase"} CGPROGRAM #pragma vertex vert #pragma fragment frag #include "UnityCG.cginc" #include "Lighting.cginc" sampler2D _MainTex; float4 _MainTex_ST; fixed4 _Color; sampler2D _BumpMap; float4 _BumpMap_ST; float _BumpScale; fixed4 _Specular; float _Gloss; struct appdata { float4 vertex : POSITION; float4 uv : TEXCOORD0; float3 normal:NORMAL; float4 tangent:TANGENT; }; struct v2f { float4 uv : TEXCOORD0; float4 vertex : SV_POSITION; float3 lightDir:TEXCOORD1; float3 viewDir:TEXCOORD2; }; v2f vert (appdata v) { v2f o; //将顶点位置转化到剪切空间 o.vertex = UnityObjectToClipPos(v.vertex); //计算主纹理的缩放和平移值 o.uv.xy = TRANSFORM_TEX(v.uv, _MainTex); //计算法线的缩放和平移值 o.uv.zw= TRANSFORM_TEX(v.uv, _BumpMap); //float3 binormal=cross(normalize(v.normal),normalize(v.tangent.xyz))*v.tangent.w; //float3x3 rotation=float3(v.tangent.xyz,binormal,v.normal); //与上面注释代码功能相同,直接计算得到rotation变换矩阵 TANGENT_SPACE_ROTATION; //获取模型空间下的光照方向 o.lightDir=mul(rotation,ObjSpaceLightDir(v.vertex)).xyz; //获取模型空间下的视角方向 o.viewDir=mul(rotation,ObjSpaceViewDir(v.vertex)).xyz; return o; } fixed4 frag (v2f i) : SV_Target { fixed3 tangentLightDir=normalize(i.lightDir); fixed3 tangentViewDir=normalize(i.viewDir); //解析法线贴图的纹理信息 fixed4 pcakedNormal=tex2D(_BumpMap,i.uv.zw); fixed3 tangentNormal; tangentNormal=UnpackNormal(pcakedNormal); //改变凹凸程度 tangentNormal.xy*=_BumpScale; tangentNormal.z=sqrt(1.0-saturate(dot(tangentNormal.xy,tangentNormal.xy))); //材质反射率 fixed3 albedo=tex2D(_MainTex, i.uv).rgb*_Color.rgb; fixed3 ambient=UNITY_LIGHTMODEL_AMBIENT.xyz*albedo; fixed3 diffuse=_LightColor0.rgb * albedo*max(0,dot(tangentNormal,tangentLightDir)); fixed3 halfDir=normalize(tangentLightDir+tangentViewDir); fixed3 specular=_LightColor0.rgb*_Specular.rgb*pow(max(0,dot(tangentNormal,halfDir)),_Gloss); return fixed4(ambient+diffuse+specular,1.0); } ENDCG } }}
二、在世界空间下,法线纹理映射的shader实现
1、基本思想
在顶点着色器中计算从切线空间到世界空间的变换矩阵,并把它传递个片元着色器。变换矩阵的计算可以有顶点的切线、副切线和法线在世界空间下的表示来得到。最后我们只需要在片元着色器中把法线纹理中的法线方向从切线空间变换到世界空间下即可。
2、源码
// Upgrade NOTE: replaced ‘_Object2World’ with ‘unity_ObjectToWorld’
Shader “Custom/Texture/WorldNormalTextute”
{
Properties
{
_MainTex (“Texture”, 2D) = “white” {}
_Color(“Color”,Color)=(1,1,1,1)
_BumpMap(“NormalMap”,2D)=”white”{}
_BumpScale(“BumpScale”,Float)=1.0
_Specular(“Specular”,Color)=(1,1,1,1)
_Gloss(“Gloss”,Range(8.0,256))=20
}
SubShader
{
Tags { “RenderType”=”Opaque” }
LOD 100
Pass { Tags{"LightMode"="ForwardBase"} CGPROGRAM #pragma vertex vert #pragma fragment frag #include "UnityCG.cginc" #include "Lighting.cginc" sampler2D _MainTex; float4 _MainTex_ST; fixed4 _Color; sampler2D _BumpMap; float4 _BumpMap_ST; float _BumpScale; fixed4 _Specular; float _Gloss; struct appdata { float4 vertex : POSITION; float4 uv : TEXCOORD0; float3 normal:NORMAL; float4 tangent:TANGENT; }; struct v2f { float4 uv : TEXCOORD0; float4 vertex : SV_POSITION; float4 TtoW1:TEXCOORD1; float4 TtoW2:TEXCOORD2; float4 TtoW3:TEXCOORD3; }; v2f vert (appdata v) { v2f o; o.vertex = UnityObjectToClipPos(v.vertex); o.uv.xy = TRANSFORM_TEX(v.uv, _MainTex); o.uv.zw= TRANSFORM_TEX(v.uv, _BumpMap); float3 worldPos=mul(unity_ObjectToWorld,v.vertex); float3 worldNormal=UnityObjectToWorldNormal(v.normal); float3 worldTangent=UnityObjectToWorldDir(v.tangent.xyz); float3 worldBinormal=cross(worldNormal,worldTangent)*v.tangent.w; o.TtoW1=float4(worldTangent.x,worldBinormal.x,worldNormal.x,worldPos.x); o.TtoW2=float4(worldTangent.y,worldBinormal.y,worldNormal.y,worldPos.y); o.TtoW3=float4(worldTangent.z,worldBinormal.z,worldNormal.z,worldPos.z); return o; } fixed4 frag (v2f i) : SV_Target { float3 worldPos=fixed3(i.TtoW1.w,i.TtoW2.w,i.TtoW3.w); fixed3 lightDir=normalize(UnityWorldSpaceLightDir(worldPos)); fixed3 viewDir=normalize(UnityWorldSpaceViewDir(worldPos)); fixed3 bump=UnpackNormal(tex2D(_BumpMap,i.uv.zw)); bump.xy*=_BumpScale; bump.z=sqrt(1.0-saturate(dot(bump.xy,bump.xy))); bump=normalize(half3(dot(i.TtoW1.xyz,bump),dot(i.TtoW2.xyz,bump),dot(i.TtoW3.xyz,bump))); fixed3 albedo=tex2D(_MainTex, i.uv).rgb*_Color.rgb; fixed3 ambient=UNITY_LIGHTMODEL_AMBIENT.xyz*albedo; fixed3 diffuse=_LightColor0.rgb * albedo*max(0,dot(bump,lightDir)); fixed3 halfDir=normalize(lightDir+viewDir); fixed3 specular=_LightColor0.rgb*_Specular.rgb*pow(max(0,dot(bump,halfDir)),_Gloss); return fixed4(ambient+diffuse+specular,1.0); } ENDCG }}
}
阅读全文
0 0
- UnityShader案例(五)——法线纹理
- UnityShader案例(四)——单张纹理
- UnityShader案例(六)——渐变纹理
- UnityShader案例(七)——遮罩纹理
- UnityShader案例(一)——漫反射
- UnityShader初级篇——单张纹理
- UnityShader初级篇——渐变纹理
- UnityShader案例(八)——透明效果(1)
- UnityShader案例(九)——透明效果(2)
- UnityShader案例(二)——Phong高光反射
- UnityShader案例(三)——BlinnPhong高光反射
- UnityShader初级篇——遮罩纹理
- Shader小常识之——模型法线和法线纹理是如何工作的
- UnityShader入门精要学习笔记(二十一):深度和法线纹理
- UnityShader官方案例讲解——SurfaceShader(1)
- UnityShader官方案例讲解——SurfaceShader(2)
- UnityShader官方案例讲解——SurfaceShader(3)
- UnityShader官方案例讲解——SurfaceShader(4)
- Android开发,Kotlin的了解与学习(二)-----Kotlin数据类型与Extensions的学习
- JQuery操作简单html组件
- oracle查询问题
- Linux-线程(一)创建 终止 等待线程
- 第四章 SpringMVC--Controller接口控制器详解(5)
- UnityShader案例(五)——法线纹理
- [Leetcode] 206. Reverse Linked List 解题报告
- 多线程系列之创建线程
- 用nginx最代理请求总是 required is not finished yet
- 自定义一个strcmp函数。 函数原型为 strcmo(char *p1,char *p2);
- 2017android面试总结
- js--异步加载
- EditText常用样式和方法小结
- java.lang.ClassCastException: com.bumptech.glide.request.target.SquaringDrawable cannot be cast to a