Unity3D shader 实现圆角矩形和圆形区域

    可以使用Marc Edwards的超椭圆方程,直接计算出圆角矩形和圆形的像素区域,并且可以使用参数调节形状的区域。


Unity3D shader 实现圆角矩形和圆形区域_第1张图片


   x,y 是坐标,5是弧度的程度越大越接近矩形,60是半径系数,shader中需要归一化转换。 利用此公式实现shader如下:

   

Shader "Custom/Vignette" 
{
	Properties 
	{
	    _fade      ("Opacity",   float) = 1
		_intensity ("Intensity", float) = 1
		_offsetX   ("OffsetX",   float) = 0
		_offsetY   ("OffsetY",   float) = 0
		_width     ("Width",     float) = 1
		_height    ("Height",    float) = 1
		_ellipse   ("Ellipse",   float) = 4
		_fuzzy     ("Fuzzy",     float) = 1
	}
	
	SubShader 
	{ 
		Tags 
		{ 
		  "QUEUE"="Transparent" 
		  "RenderType"="Transparent" 
		}
		Pass 
		{
			ZWrite Off
			Blend SrcAlpha OneMinusSrcAlpha

			CGPROGRAM
			#pragma vertex vert
			#pragma fragment frag
			#include "UnityCG.cginc"

			struct v2f 
			{
				float4 vertex : POSITION;
				float2  texcoord : TEXCOORD0;
			};

			struct appdata_t 
			{
				float4 vertex : POSITION;
			};

			float _offsetX;
			float _offsetY;
			float _fade;
			float _intensity;
			float _width;
			float _height;
			float _ellipse;
			float _fuzzy;
			
			v2f vert(in appdata_t v) 
			{
				float4 tmpvar_2 = mul(UNITY_MATRIX_MVP, v.vertex);
				float2 tmpvar_3 = (tmpvar_2.xy / tmpvar_2.w);
				
				float2 tmpvar_1;
				tmpvar_1.x = (tmpvar_3.x + _offsetX);
				tmpvar_1.y = (tmpvar_3.y + _offsetY) * 2.5;
				
				v2f o;
				o.vertex   = tmpvar_2;
				o.texcoord = tmpvar_1;
				
				return o;
			}

			half4 frag(v2f IN) : COLOR
			{
				half4 col;
				
				col.xyz   = half3(0.0, 0.0, 0.0);
				
				col.w = clamp
				        (
							pow(abs(IN.texcoord.x / 0.5) * _width,  _ellipse) + 
							pow(abs(IN.texcoord.y / 0.5) * _height, _ellipse),
							0.0, 
							1.0
						);
						
				if (col.w < 1.0f)
				{	
					col.w *= _fade * _intensity * _fuzzy;
				}				
				else
				{
					col.w *= _fade * _intensity;
				}				
				
				
				return  col;
			}

			ENDCG

		}
	}
	
	FallBack "Diffuse"
}

    效果如下:


Unity3D shader 实现圆角矩形和圆形区域_第2张图片      Unity3D shader 实现圆角矩形和圆形区域_第3张图片


    主要的思路是,利用公式计算式高亮部分的透明度,其他的区域全不透明。shader提供了各种参数可以调价,大小明亮度和边缘的过度效果。


    如果需要去圆角化一个已有的矩形纹理,需要利用公式算出的区域的纹理数据不变。不在区域内的部分不显示即可到达剪裁的目的。


你可能感兴趣的:(Unity3D)