您的位置:首页 > 移动开发 > Unity3D

Unity3D ShaderLab 使用贴图对模型的高光进行遮罩

2015-01-18 22:15 344 查看
Unity3D ShaderLab 使用贴图对模型的高光进行遮罩

前面研究了高光效果的实现,再说说现很多游戏用到的高光贴图技术,因为它可以让3D美工更容易控制最终的视觉效果。

这也就为我们提供了另外的方式,我们可以在同一个着色器上实现垫型表面和光亮表面,也可以使用贴图来控制镜面高光的范围或者高光强度,

以实现一个表面是广泛的镜面高光而另一面确是细小的高光。

新建一个shader,一个材质球。老规矩双击shader脚本,在编辑器中编写代码吧。

1.Properties:

Properties {

_MainTint("Diffuse Tint",Color) = (1,1,1,1)

_MainTex ("Base (RGB)", 2D) = "white" {}

_SpecularColor("Specular Tint",Color)=(1,1,1,1)

_SpecularMask("Specular Texture",2D)="white"{}

_SpecularPower("Specular Power",Range(1,100))=3

}


2.SubShader中修改CGPROGRAM,加入输出结构体SurfaceMyOutput,修改Input结构体:

CGPROGRAM

#pragma surface surf TexPhong

sampler2D _MainTex;

float4 _MainTint;

float4 _SpecularColor;

sampler2D _SpecularMask;

float _SpecularPower;

struct SurfaceMyOutput{

fixed3 Albedo;

fixed3 Normal;

fixed3 Emission;

fixed3 SpecularColor;

half Specular;

fixed Gloss;

fixed Alpha;

};

struct Input {

float2 uv_MainTex;

float2 uv_SpecularMask;

};


3.实现自定义光照模型LightingTexPhong

inline fixed4 LightingTexPhong(SurfaceMyOutput s,fixed3 lightDir,half3 viewDir, fixed atten){

float diff = dot(s.Normal,lightDir);

float3 reflection = normalize(3.0*s.Normal*diff-lightDir);

float spec = pow(max(0,dot(reflection,viewDir)), _SpecularPower)*s.Specular;

float3 finalSpec = s.SpecularColor * spec*_SpecularColor.rgb;

fixed4 c;

c.rgb = (s.Albedo*_LightColor0.rgb*diff)+(_LightColor0.rgb*finalSpec);

c.a = s.Alpha;

return c;

}


4.修改surf函数

void surf (Input IN, inout SurfaceMyOutput o) {

float4 c = tex2D (_MainTex, IN.uv_MainTex)*_MainTint;

float4 specMask = tex2D(_SpecularMask,IN.uv_SpecularMask)*_SpecularColor;

o.Albedo = c.rgb;

o.Specular = specMask.r;

o.SpecularColor = specMask.rgb;

o.Alpha = c.a;

}


修改完毕后,返回unity设置参数,最终效果如下。



在上面的代码编写过程中,我们需要将表面函数的信息传递给给光照函数,因为我们在光照函数内部不能得到一个物体表面的uv,

所以我们需要通过input结构体来访问数据,唯一途径就是使用surf函数,为了建立数据关系,我们自定义了结构体SurfaceMyOutput,

用这个结构体作为容器存储表面着色器中所有最终数据。这样光照函数和surf函数都可以访问它的内部数据。

然后,我们告诉surf函数和光照函数使用output结构体是我们自定义的SurfaceMyOutput结构体,而不是着色器内置的。

最后我们只需要使用tex2D函数就可以访问纹理讯息返回值,直接传递给SurfaceMyOutput结构体,完成了这些,我们就可以在光照函数中访问讯息纹理。

code start --------------------------------------------------------------------------

Shader "91YGame/TexPhong" {
Properties {
_MainTint("Diffuse Tint",Color) = (1,1,1,1)
_MainTex ("Base (RGB)", 2D) = "white" {}

_SpecularColor("Specular Tint",Color)=(1,1,1,1)
_SpecularMask("Specular Texture",2D)="white"{}
_SpecularPower("Specular Power",Range(0.5,100))=3
}
SubShader {
Tags { "RenderType"="Opaque" }
LOD 200

CGPROGRAM
#pragma surface surf TexPhong

sampler2D _MainTex;
float4 _MainTint;
float4 _SpecularColor;
sampler2D _SpecularMask;
float _SpecularPower;

struct SurfaceMyOutput{
fixed3 Albedo;
fixed3 Normal;
fixed3 Emission;
fixed3 SpecularColor;
half Specular;
fixed Gloss;
fixed Alpha;
};

struct Input {
float2 uv_MainTex;
float2 uv_SpecularMask;
};

inline fixed4 LightingTexPhong(SurfaceMyOutput s,fixed3 lightDir,half3 viewDir, fixed atten){
float diff = dot(s.Normal,lightDir);
float3 reflection = normalize(3.0*s.Normal*diff-lightDir);

float spec = pow(max(0,dot(reflection,viewDir)), _SpecularPower)*s.Specular;
float3 finalSpec = s.SpecularColor * spec*_SpecularColor.rgb;

fixed4 c;
c.rgb = (s.Albedo*_LightColor0.rgb*diff)+(_LightColor0.rgb*finalSpec);
c.a = s.Alpha;
return c;
}

void surf (Input IN, inout SurfaceMyOutput o) {
float4 c = tex2D (_MainTex, IN.uv_MainTex)*_MainTint;
float4 specMask = tex2D(_SpecularMask,IN.uv_SpecularMask)*_SpecularColor;
o.Albedo = c.rgb;
o.Specular = specMask.r;
o.SpecularColor = specMask.rgb;
o.Alpha = c.a;
}
ENDCG
}
FallBack "Diffuse"
}


code end ---------------------------------------------------------------------------
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: