Unity 水波纹Shader

今天来个炫酷的点击屏幕实现水波纹的特效,先上两张效果图。

放大下看下细节,

当然还有好多效果,自己发挥了。。。

直接上代码,注释已经写了哦

    Shader "Custom/ClickStyle" {
        Properties {
            [PerRendererData]_MainTex("Sprite Texture",2D)="white"{}
            [HideInInspector]_StartTime("Start Time",Float)=0  //起始时间,这个在Ripple中设置就行
            _AnimationTime("AnimationTime",Range(0.1,10.0))=1.5 //动画时间
            _Width("Width",Range(0.1,3.0))=0.3   //内圆和外圆形成环的宽度
            _StartWidth("Start Width",Range(0,1.0))=0.3  //内圆的默认直径,注意这里是按照uv尺寸走的,最大也就是1
            [Toggle]_isAlpha("isAlpha",Float)=1
            [Toggle]_isColorShift("isColorShift",Float)=1
            [MaterialToggle]PixelSnap("Pixe Snap",Float)=1
        }
        SubShader {

            //启用透明混合,要不然没有透明效果哦
            Tags{"Queue"="Transparent" "RenderType"="Transparent"}
            Blend SrcAlpha OneMinusSrcAlpha

            Pass
            {
                CGPROGRAM
                #pragma target 3.0
                #pragma vertex vert 
                #pragma fragment frag 
                #include "UnityCG.cginc"

                struct v2f{
                     float4 vertex:SV_POSITION;
                     float2 texcoord:TEXCOORD0;
                };
                sampler2D _MainTex;
                float _StartTime;
                float _AnimationTime;
                float _StartWidth;
                float _Width;
                float _isAlpha;
                float _isColorShift;

                v2f vert(appdata_base IN)
                {
                    v2f OUT;
                    OUT.vertex=UnityObjectToClipPos(IN.vertex);
                    OUT.texcoord=IN.texcoord;
                    return OUT;
                }

                //这个函数是用来变换颜色的,当然你可以想怎么干就怎么干,随你洛。
                //注意这个函数写在frag函数前面,要不然不识别哦
                fixed3 shift_col(fixed3 RGB,half3 shift)
                {
                    fixed3 RESULT = fixed3(RGB);
                    float VSU = shift.z*shift.y*cos(shift.x*3.14159265 / 180);
                    float VSW = shift.z*shift.y*sin(shift.x*3.14159265 / 180);

                    RESULT.x = (.299*shift.z + .701*VSU + .168*VSW)*RGB.x
                    + (.587*shift.z - .587*VSU + .330*VSW)*RGB.y
                    + (.114*shift.z - .114*VSU - .497*VSW)*RGB.z;

                    RESULT.y = (.299*shift.z - .299*VSU - .328*VSW)*RGB.x
                    + (.587*shift.z + .413*VSU + .035*VSW)*RGB.y
                    + (.114*shift.z - .114*VSU + .292*VSW)*RGB.z;

                    RESULT.z = (.299*shift.z - .3*VSU + 1.25*VSW)*RGB.x
                    + (.587*shift.z - .588*VSU - 1.05*VSW)*RGB.y
                    + (.114*shift.z + .886*VSU - .203*VSW)*RGB.z;

                    return RESULT;
                }


                fixed4 frag(v2f IN):SV_Target{
                    fixed4 color=tex2D(_MainTex,IN.texcoord);
                    float2 pos=(IN.texcoord-float2(0.5,0.5))*2; //计算像素点到中心的距离,乘以2就当做圆的直径吧。


                 //如果从中心开始向外扩散
                //  float dis=(_Time.y-_StartTime)/_AnimationTime-length(pos);

                //如果指定内径开始向外扩散 ,在Ripple类中 已经减去了开始长度的时间。所以这里就不用再减了
                float dis=(_Time.y-_StartTime)/_AnimationTime+_StartWidth-length(pos);
                    //大于最大宽度以及小于0都去掉这部分像素

                    if(dis<0||dis>_Width)
                    {
                      return fixed4(0,0,0,0);
                    }

                    //测试用
                    //if(dis < 0 )
                    //{
                    //  return fixed4(1,0,0,1);
                    //}
                    //if(dis>_Width)
                    //{
                    //  return fixed4(0,1,0,1);
                    //}

                    //如果开启了透明度渐变就让透明度进行插值
                    float alpha=1;
                    if(_isAlpha==1)
                    {
                        alpha = clamp((_Width-dis)*3,0.1,1.5);
                    }

                    fixed3 shiftColor=color;
                    if(_isColorShift==1)
                    {
                        half3 shift=half3(_Time.w*10,1,1);
                        shiftColor=shift_col(color,shift);
                    }

                    return fixed4(shiftColor,color.a*alpha);
                }


                ENDCG
            }
        }
    }

注意Shader的测试技巧哦。

C#代码如下:

        using System.Collections;
        using System.Collections.Generic;
        using UnityEngine;

        [RequireComponent(typeof(SpriteRenderer))]
        [RequireComponent(typeof(Collider2D))]
        public class Ripple : MonoBehaviour {

            SpriteRenderer mSpriteRenderer;
            Collider2D mCircleCollider;

            void Awake()
            {
                mSpriteRenderer = transform.GetComponent();
                mCircleCollider = transform.GetComponent();
            }
            // Use this for initialization
            void Start () {
                Invoke("unenbaleTrigger", 0.05f);

                mSpriteRenderer.material.SetFloat("_StartTime", Time.time);

                float animationTime = mSpriteRenderer.material.GetFloat("_AnimationTime");
                float destroyTime = animationTime;
                //需要减去起始位置所需要消耗的时间
                destroyTime -= mSpriteRenderer.material.GetFloat("_StartWidth") * animationTime;
                Destroy(transform.gameObject, destroyTime);
            }

            public void unenbaleTrigger()
            {
                mCircleCollider.enabled = false;
            }

            public void OnTriggerEnter2D(Collider2D collision)
            {

            }

        }

子健创建一个空物体,然后按照下图对应添加组件就完事

把这个对象作成预设。

再弄个点击屏幕,然后生成预设体脚本。代码如下:

    using System.Collections;
    using System.Collections.Generic;
    using UnityEngine;
    using UnityEngine.EventSystems;
    using UnityEngine.UI;

    [RequireComponent(typeof(Image))]
    public class Control : MonoBehaviour,IPointerDownHandler,IPointerUpHandler,IDragHandler {

        public GameObject prefabRippleEffect;
        public float distance;
        // Use this for initialization
        void Start () {
            distance = transform.localPosition.z - Camera.main.transform.localPosition.z;
        }

        // Update is called once per frame
        void Update () {
            CheckTap();
        }
        void CheckTap()
        {

            if (Input.GetMouseButtonDown(0))
            {

                var dian = Input.mousePosition;
                Debug.Log(dian);
                CreateNewRipple(dian);
            }

        }
        void CreateNewRipple(Vector2 pos)
        {
            Vector2 worldPos = Camera.main.ScreenToWorldPoint(new Vector3(pos.x,pos.y,distance));
            Debug.Log(worldPos);
            GameObject tem = Instantiate(prefabRippleEffect);
            tem.transform.localPosition = new Vector3(worldPos.x,worldPos.y,0);
            tem.transform.SetParent(transform,true);
        }
        public void OnPointerDown(PointerEventData eventData)
        {
           // CreateNewRipple(eventData.position);
        }

        public void OnPointerUp(PointerEventData eventData)
        {

        }

        public void OnDrag(PointerEventData eventData)
        {

        }
    }

创建一个空物体,把这个脚本挂上去即可。

不明白的地方欢迎留言交流哦。。。


附上工程链接:水波纹Shader

你可能感兴趣的:(Shader)