1. 程式人生 > >Cg入門19:Fragment shader - 片段級模型動態變色

Cg入門19:Fragment shader - 片段級模型動態變色

tin ogr 沒有 動態 erp unity data pro targe

技術分享技術分享
y值要表示範圍為[-0.5,0.5],所以語義要註意不要用Color(註意:Color 語義值範圍為[0,1](特別註意:內建的cube範圍才是【-0.5,0.5】,其它模型就不一定是這個值了噢)技術分享
技術分享
技術分享技術分享
發現頂部沒有融合掉。為了所有融合掉。我們將_Center範圍+R的大小,就所有融合了技術分享
技術分享效果例如以下:技術分享技術分享

優化:去掉if else,由於if else 可能在其它硬件上不能運行。技術分享技術分享技術分享

代碼:
Shader "Sbin/vf55" {
	Properties
	{
		_UpColor("UpColor",color) = (1,0,0,1)
		_DownColor("DownColor",color) = (0,1,0,1)
		_Center("Center",range(-0.7,0.7)) = 0
		_R("R",range(0,0.5)) = 0.2
	}

	SubShader {
	   
		pass{
			Tags{"LightMode" = "ForwardBase"}

			CGPROGRAM
			#pragma vertex vert
			#pragma fragment frag

			#include "UnityCG.cginc"
			#include "Lighting.cginc"

			struct v2f{
				float4 pos : POSITION;
				float y:TEXCOORD0;
			};

			float4 _UpColor;
			float4 _DownColor;
			float _Center; 
			float _R; 

			v2f vert(appdata_base v){
				v2f o;
				o.pos = mul(UNITY_MATRIX_MVP,v.vertex);
				o.y = v.vertex.y;

				return o;
			}
			
			fixed4 frag(v2f v):COLOR
			{
				//方法1:
				//=========start============
				/*if(v.y > _Center + _R)
				{
					return _UpColor;
				}
				else if(v.y>_Center && v.y <=_Center + _R)
				{
					float d = v.y - _Center;
					d = saturate(1-d/_R - 0.5);
					return lerp(_UpColor,_DownColor,d);
				}
				else if(v.y<=_Center && v.y > _Center - _R)
				{
					float d = _Center - v.y;
					d = saturate(1-d/_R-0.5);
					return lerp(_DownColor,_UpColor,d);
				}
				else
				{
					return _DownColor;
				}*/
				//==========end===========
				//方法2:
				//=========start==========
				float d = v.y - _Center;//融合帶
				float s = abs(d);
				d = d/s;//正負值分別描寫敘述上半部分和下半部分,取值1和-1

				float f = s/_R;	//範圍>1:表示上下部分;範圍<1:表示融合帶
				f = saturate(f);
				d *= f;//表示所有[-1,1];範圍>1:表示上部分;範圍<1:表示融合帶;範圍<-1:表示下部分
				
				d = d/2+0.5;//將範圍控制到[0,1],由於顏色值返回就是[0,1]
				return lerp(_UpColor,_DownColor,d);
				//==========end===========
			}

			ENDCG
		}
	} 
}

終於效果:技術分享
技術分享

Cg入門19:Fragment shader - 片段級模型動態變色