1. 程式人生 > 程式設計 >unity實現透明水波紋扭曲

unity實現透明水波紋扭曲

本文例項為大家分享了unity實現透明水波紋扭曲的具體程式碼,供大家參考,具體內容如下

需要掛一個攝像機把指令碼掛在一個物體上

可隨意在物體上面點選

shader:

Shader "Unlit/Water"
{
 Properties
 {
 _MainTex ("Texture",2D) = "white" {}
 _WaterUV("WaterUV",2D)="while"{}
 _WaterIntensity("WaterIntensity",float)=500
 }
 SubShader
 {

 GrabPass{
 Name "BASE"
 Tags { "Mode" = "Always" }
 }

 Tags { "Queue"="Transparent+100" "RenderType"="Transparent" }
 

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

 struct appdata
 {
 float4 vertex : POSITION;
 float2 uv : TEXCOORD0;
 float3 normal:Normal;
 };

 struct v2f
 {
 float2 uv : TEXCOORD0;
 float4 grabuv:TEXCOORD1;
 float4 vertex : SV_POSITION;
 float3 normal:Normal;
 };

 sampler2D _MainTex;
 float4 _MainTex_ST;
 sampler2D _GrabTexture;
 sampler2D _WaterUV;
 float4 _GrabTexture_TexelSize;
 float _WaterIntensity;

 v2f vert (appdata v)
 {
 v2f o;
 o.vertex = mul(UNITY_MATRIX_MVP,v.vertex);
 o.uv = TRANSFORM_TEX(v.uv,_MainTex);

 UNITY_TRANSFER_FOG(o,o.vertex);
 o.grabuv=ComputeGrabScreenPos(o.vertex);

 o.normal=v.normal;


 return o;
 }
 
 fixed4 frag (v2f i) : SV_Target
 {
 fixed4 col = tex2D(_MainTex,i.uv);
 float2 uv=tex2D(_WaterUV,i.uv).xy;

  uv= (uv-0.5)*2;

 float2 offset_ =uv * _GrabTexture_TexelSize.xy*_WaterIntensity;
 
 float4 grabuv=i.grabuv;
 grabuv.xy+=sin(offset_);

 fixed4 grabCol=tex2Dproj(_GrabTexture,UNITY_PROJ_COORD(grabuv));
 
 
 return col*grabCol;
 }
 ENDCG
 }
 }
}

C#:

using UnityEngine;
using System.Collections;
using System.Threading;

public class Water : MonoBehaviour
{


 public int m_texHeight = 512;

 public int m_texWidth = 512;

 public Texture2D m_waterTexture = null;

 private Material m_waterMatrial;

 private float[,] waveA;
 private float[,] waveB;

 public float decrement=0.025f;

 public Camera m_waterCam;

 private int time;
 void Start()
 {

  m_waterTexture = new Texture2D(m_texWidth,m_texHeight,TextureFormat.RGBA32,false);

  m_waterMatrial = GetComponent<MeshRenderer>().material;

  waveA = new float[m_texWidth,m_texHeight];
  waveB = new float[m_texWidth,m_texHeight];
 
  m_waterMatrial.SetTexture("_WaterUV",m_waterTexture);

  allColor = new Color[m_texWidth* m_texHeight];

  Thread t = new Thread(new ThreadStart(ThreadWave));
  t.Start();

  // m_waterMatrial.SetTexture("_MainTex",m_waterTexture);
 }

 public void PopWater(Vector2 pos)
 {
  float x=pos.x;
  float y=pos.y;
  waveA[(int)(m_texWidth * x),(int)(m_texHeight * y)] = 1;
  //waveA[(int)(m_texWidth * x) - 1,(int)(m_texHeight * y) ] = 1;
  //waveA[(int)(m_texWidth * x) + 1,(int)(m_texHeight * y) ] = 1;
  //waveA[(int)(m_texWidth * x),(int)(m_texHeight * y) - 1] = 1;
  //waveA[(int)(m_texWidth * x),(int)(m_texHeight * y) + 1] = 1;
  //waveA[(int)(m_texWidth * x) - 1,(int)(m_texHeight * y) - 1] = 1;
  //waveA[(int)(m_texWidth * x) - 1,(int)(m_texHeight * y) + 1] = 1;
  //waveA[(int)(m_texWidth * x) + 1,(int)(m_texHeight * y) - 1] = 1;
  //waveA[(int)(m_texWidth * x) + 1,(int)(m_texHeight * y) + 1] = 1;
 }
 public void PopWater()
 {
  waveA[(int)(m_texWidth / 2),(int)(m_texHeight / 2)] = 1;
  waveA[(int)(m_texWidth / 2) - 1,(int)(m_texHeight / 2)] = 1;
  waveA[(int)(m_texWidth / 2) + 1,(int)(m_texHeight / 2)] = 1;
  waveA[(int)(m_texWidth / 2),(int)(m_texHeight / 2) - 1] = 1;
  waveA[(int)(m_texWidth / 2),(int)(m_texHeight / 2) + 1] = 1;
  waveA[(int)(m_texWidth / 2) - 1,(int)(m_texHeight / 2) - 1] = 1;
  waveA[(int)(m_texWidth / 2) - 1,(int)(m_texHeight / 2) + 1] = 1;
  waveA[(int)(m_texWidth / 2) + 1,(int)(m_texHeight / 2) - 1] = 1;
  waveA[(int)(m_texWidth / 2) + 1,(int)(m_texHeight / 2) + 1] = 1;
 }


 void Update()
 {
  ComputeWave();


  if (Input.GetMouseButtonDown(0))
  {
   RaycastHit rh;
   if (Physics.Raycast(m_waterCam.ScreenPointToRay(Input.mousePosition),out rh))
   {
    

    PopWater(rh.textureCoord);
   }

  }

  time = (int)Time.deltaTime * 1000;
 
 }

 void OnDestroy()
 {
  f = false;

 }

 bool f = true;
 void ThreadWave()
 {
  while (f)
  {
   Thread.Sleep(time);
   for (int w = 1; w < m_texWidth - 1; w++)
   {

    for (int h = 1; h < m_texHeight - 1; h++)
    {
     waveB[w,h] =
      (waveA[w - 1,h] +
      waveA[w + 1,h] +
      waveA[w,h - 1] +
      waveA[w,h + 1] +
      waveA[w - 1,h - 1] +
      waveA[w + 1,h - 1] +
      waveA[w - 1,h + 1] +
      waveA[w + 1,h + 1]) / 4 - waveB[w,h];


     float value = waveB[w,h];
     if (value > 1)
     {
      waveB[w,h] = 1;
     }

     if (value < -1)
     {
      waveB[w,h] = -1;
     }


     if (value > -0.0001 && value < 0.0001)
     {
      waveB[w,h] = 0;
     }


     float offset_u = (waveB[w - 1,h] - waveB[w + 1,h]) / 2;

     float offset_v = ((waveB[w,h - 1]) - waveB[w,h + 1]) / 2;


     float r = offset_u / 2 + 0.5f;
     float g = offset_v / 2 + 0.5f;
     Color c = new Color(r,g,0);
     

     waveB[w,h] -= waveB[w,h] * decrement;
     allColor[w + m_texWidth * h] = c;


    }
   }

   float[,] temp;
   temp = waveA;
   waveA = waveB;
   waveB = temp;
  }

 }

 private Color[] allColor;

 void ComputeWave()
 {
  m_waterTexture.SetPixels(allColor);
  m_waterTexture.Apply();



 }

}

效果圖:

unity實現透明水波紋扭曲

unity實現透明水波紋扭曲

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。