从零开始的RPG制作(外传1-一个不是很实用的水面行走效果)

效果

虽然通过参数调试,可以得到不错的水面行走效果,但是CPU计算量也太大了了点。唉~
这里的实现原理,是通过一张Texture2D图,将UV波动的信息当成颜色记录在Texture2D图上,然后通过shader的tex2D函数获得颜色(也就是获得了UV),将这部分和原始UV叠加就能获得人物行走水面的效果。因为Texture2D贴图越大就越耗,计算量也是指数上升,自欺欺人的放入线程计算,不过看着过高的cpu占用,只能舍弃了,唉~,不过将Texture2D贴图限制的小一点,说不定就能吧。还有一个方案就是放在computer shader中计算,下面放上代码,注解很详细了。
脚本

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

public class WaterWave:MonoBehaviour {

    // Use this for initialization
    int waveWidth = 512;
    int waveHeigth = 512;
    public int waveWidthMax = 512;
    public int waveHeigthMax = 512;
    int planeW = 5;
    int planeH = 5;
    UnityEngine.Color[] colorBuff;
    [Range(0, 1)]
    public float power = 0.2f;
    [Range(0, 0.2f)]
    public float weak = 0.02f;
    float[,] waveA;
    float[,] waveB;
    Texture2D tex_uv;

    void Start() {
        waveA = new float[waveWidth, waveWidth];
        waveB = new float[waveHeigth, waveHeigth];
        tex_uv = new Texture2D(waveWidth, waveHeigth);
        GetComponent<Renderer>().material.SetTexture("_waveTex", tex_uv);
        colorBuff = new UnityEngine.Color[waveWidth * waveHeigth];
        isTure = true;
        Thread tf = new Thread(new ThreadStart(computeWave));
        tf.Start();
        UpdateManges.add_playerEventList_(waveUpdate);
        //computeWave();
    }

    // Update is called once per frame
    bool ISPLAY = false;
    int thisTime = 0;
    public Vector3 vt3;//人物脚部落地的世界坐标
    void waveUpdate() {
        thisTime = (int)(Time.deltaTime * 1000);
        tex_uv.SetPixels(colorBuff);
        Debug.Log(colorBuff[0][0]);
        tex_uv.Apply();
        /*if (Input.GetMouseButtonDown(0)) {
            ISPLAY = true;
        } else if (Input.GetMouseButtonUp(0)) {
            ISPLAY = false;
        }
        if (ISPLAY) {
            Vector3 vt = Input.mousePosition;
            Ray r = Camera.main.ScreenPointToRay(vt);
            RaycastHit hitt = new RaycastHit();
            if (Physics.Raycast(r, out hitt)) {
                Vector3 localPos = hitt.collider.transform.InverseTransformPoint(hitt.point);
                localPos.x = planeW - (localPos.x + planeW) / 2;
                localPos.z = planeH - (localPos.z + planeH) / 2;
                putpop(localPos.x / planeH * waveWidth, localPos.z / planeH * waveHeigth, power);
            }
        }*/
        if (vt3 != Vector3.zero) {
            Vector3 localPos = transform.InverseTransformPoint(vt3);
            localPos.x = planeW - (localPos.x + planeW) / 2;
            localPos.z = planeH - (localPos.z + planeH) / 2;
            putpop(localPos.x / planeH * waveWidth, localPos.z / planeH * waveHeigth, power);
        }
    }

    void putpop(float x, float y, float power) {
        int radius = 3;
        int _x = (int)x < radius + 1 ? radius + 1 : (int)x >= waveWidth - radius - 1 ? waveWidth - radius - 1 : (int)x;
        int _y = (int)y < radius + 1 ? radius + 1 : (int)y >= waveHeigth - radius - 1 ? waveHeigth - radius - 1 : (int)y;

        float dist;
        for (int i = -radius; i <= radius; i++) {
            for (int j = -radius; j <= radius; j++) {
                dist = Mathf.Sqrt(i * i + j * j);
                if (dist < radius) {
                    waveA[_x + i, _y + j] = power*Mathf.Cos(dist * Mathf.PI / radius);//一个范围的都添加力量
                }
            }
        }
    }

    void computeWave() {
        while (isTure) {
            for (int w = 1; w < waveWidthMax - 1; w++) {
                for (int h = 1; h < waveHeigthMax - 1; h++) {
                    waveB[w, h] = (waveA[w - 1, h] +//左
                        waveA[w - 1, h + 1] +//左上
                        waveA[w, h + 1] +//上
                        waveA[w + 1, h + 1] +//右上
                        waveA[w + 1, h] +//右
                        waveA[w + 1, h - 1] +//右下
                        waveA[w, 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;
                    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 + 1) / 2;//0-1
                    float g = (offset_v + 1) / 2;//0-1

                    colorBuff[waveWidth * h + w] = new UnityEngine.Color(r, g, 0);//将UV信息写入颜色,最后通过tex2d来获取UV信息

                    waveB[w, h] -= waveB[w, h] * weak;//波纹衰减衰减
                }
            }

            /*for (int w = waveWidth - waveWidthMax; w < waveWidth - 1; w++) {
                for (int h = waveHeigth - waveHeigthMax; h < waveHeigth - 1; h++) {
                    colorBuff[waveWidth * h + w - (waveWidth - waveWidthMax)] = new UnityEngine.Color(0, 0, 0);
                    waveB[w, h] = 0;
                }
            }*/

            float[,] temp = waveA;
            waveA = waveB;
            waveB = temp;
            Thread.Sleep(thisTime);
        }
    }
    bool isTure = false;
    private void OnDestroy() {
        UpdateManges.sub_playerEventList_(waveUpdate);
        isTure = false;
    }

}

shader

Shader "Unlit/WaterWave"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
        _F("平静状态下的水扰动周期",range(0,10)) = 1
        _P("平静状态下的水平移范围", range(0, 10)) = 1
        wave("波涛起伏深度",range(0,1)) = 0.5
    }
    SubShader
    {
        Tags { "RenderType"="Opaque" }
        LOD 100

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

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

            struct v2f
            {
                float2 uv : TEXCOORD0;
                float4 vertex : SV_POSITION;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;
            sampler2D _waveTex;
            float _F;
            float _P;
            float wave;
            v2f vert (appdata v)
            {
                v2f o;
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                return o;
            }
            
            fixed4 frag (v2f i) : SV_Target
            {
                i.uv += 0.01 *sin(i.uv * 2 * 3.14*_F + _Time.y*_P);
                // sample the texture
                float2 waveuv = tex2D(_waveTex, i.uv).xy; //0-1
                waveuv = waveuv * wave;
                i.uv += waveuv;

                fixed4 col = tex2D(_MainTex, i.uv);
                return col;
            }
            ENDCG
        }
    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,372评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,368评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,415评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,157评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,171评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,125评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,028评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,887评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,310评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,533评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,690评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,411评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,004评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,812评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,693评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,577评论 2 353