Unity URP 如何写基础的几何着色器

这是使用几何着色器在点中心生成一个点并根据这个点把原本的面片分成三个三角形的操作。

对于几何着色器构造相对简单,网上的信息也相对较多,需要注意的点就是需要提供一个新的数据结构供几何着色器输出,因为几何着色器在顶点之后,片元之前,所以结构体就需要模型输入的结构体、顶点输出的结构体、几何输出的结构体。

下面是完整代码

Shader "Kerzh/GeoShaderTest"
{
    Properties
    {
        _Length("Length",Float) = 1
    }
    SubShader
    {
        Tags {"LightMode" = "UniversalForward"}
        LOD 100

        Pass
        {
            Cull Off
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma geometry geom

            #include "UnityCG.cginc"

            //cpu 传到 vs
            struct appdata
            {
                float4 posOS : POSITION;
                float2 uv : TEXCOORD0;
                float4 normalOS : NORMAL;
            };

            //vs 传到 gs
            struct v2g
            {
                float4 posOS    : TEXCOORD0;
                float2 uv       : TEXCOORD1;
                float3 normalOS : TEXCOORD2;
            };

            //gs 传到 fs
            struct g2f
            {
                float4 posCS    : SV_POSITION;
                float2 uv       : TEXCOORD0;
                float3 normalWS : TEXCOORD1;
                float3 posWS    : TEXCOORD2;
                float  dis      : TEXCOORD3;
            };
            
            float _Length;

            v2g vert (appdata input)
            {
                v2g output;
                output.uv = input.uv;
                output.normalOS = input.normalOS;
                output.posOS = input.posOS;
                return output;
            }

            //最大的顶点数,这里比如你要生成三个三角形面片,那么一个面片需要三个顶点,就是9个顶点
            [maxvertexcount(9)]
            void geom (triangle v2g input[3],inout TriangleStream<g2f> triStream)
            {
                g2f center;  //  中间那个点
                center.uv = (input[0].uv + input[1].uv + input[2].uv)/3.0;//  计算中间点的uv
                float3 centerNormalOS = (input[0].normalOS + input[1].normalOS + input[2].normalOS)/3.0;
                center.normalWS = UnityObjectToWorldNormal(centerNormalOS );//  计算中间点的法线
                float3 centerPosOS = (input[0].posOS + input[1].posOS + input[2].posOS)/3.0;//  计算中间点的位置
                centerPosOS = centerPosOS + centerNormalOS *_Length;//  沿着中心点法线根据噪声及系数位移中心点

                //中心点后期所需数据
                center.posWS = mul(unity_ObjectToWorld, centerPosOS);
                center.posCS = UnityObjectToClipPos(centerPosOS);
                center.dis = 1;
                
                g2f output[3];  //  这里是原样把输入三角形的三个点放到了这里面,只是为了下面取得方便
                for(int i=0;i<3;i++)
                {
                    g2f p0;
                    p0.uv = input[i].uv;
                    p0.normalWS = UnityObjectToWorldNormal( input[i].normalOS);
                    p0.posWS = mul(unity_ObjectToWorld, input[i].posOS);
                    p0.posCS = UnityObjectToClipPos(input[i].posOS );
                    p0.dis = 0;
                    output[i] = p0;
                }

                //  根据这三个点分别和中心点制造三角形输出
                triStream.RestartStrip(); //  重新开始一个新的三角形
                triStream.Append(output[1]);
                triStream.Append(center);
                triStream.Append(output[0]);

                triStream.RestartStrip();  //  重新开始一个新的三角形
                triStream.Append(output[2]);
                triStream.Append(center);
                triStream.Append(output[1]);
                
                triStream.RestartStrip(); //  重新开始一个新的三角形
                triStream.Append(output[0]);
                triStream.Append(center);
                triStream.Append(output[2]);
            }
            
            float4 frag (g2f i) : SV_Target
            {
                return i.dis;
                return float4(i.uv,0,0);
            }
            ENDCG
        }
    }
}

关于其中几何着色器的部分

//最大的顶点数,这里比如你要生成三个三角形面片,那么一个面片需要三个顶点,就是9个顶点
            [maxvertexcount(9)]
            void geom (triangle v2g input[3],inout TriangleStream<g2f> triStream)
            {
                g2f center;  //  中间那个点
                center.uv = (input[0].uv + input[1].uv + input[2].uv)/3.0;//  计算中间点的uv
                float3 centerNormalOS = (input[0].normalOS + input[1].normalOS + input[2].normalOS)/3.0;
                center.normalWS = UnityObjectToWorldNormal(centerNormalOS );//  计算中间点的法线
                float3 centerPosOS = (input[0].posOS + input[1].posOS + input[2].posOS)/3.0;//  计算中间点的位置
                centerPosOS = centerPosOS + centerNormalOS *_Length;//  沿着中心点法线根据噪声及系数位移中心点

                //中心点后期所需数据
                center.posWS = mul(unity_ObjectToWorld, centerPosOS);
                center.posCS = UnityObjectToClipPos(centerPosOS);
                center.dis = 1;
                
                g2f output[3];  //  这里是原样把输入三角形的三个点放到了这里面,只是为了下面取得方便
                for(int i=0;i<3;i++)
                {
                    g2f p0;
                    p0.uv = input[i].uv;
                    p0.normalWS = UnityObjectToWorldNormal( input[i].normalOS);
                    p0.posWS = mul(unity_ObjectToWorld, input[i].posOS);
                    p0.posCS = UnityObjectToClipPos(input[i].posOS );
                    p0.dis = 0;
                    output[i] = p0;
                }

                //  根据这三个点分别和中心点制造三角形输出
                triStream.RestartStrip(); //  重新开始一个新的三角形
                triStream.Append(output[1]);
                triStream.Append(center);
                triStream.Append(output[0]);

                triStream.RestartStrip();  //  重新开始一个新的三角形
                triStream.Append(output[2]);
                triStream.Append(center);
                triStream.Append(output[1]);
                
                triStream.RestartStrip(); //  重新开始一个新的三角形
                triStream.Append(output[0]);
                triStream.Append(center);
                triStream.Append(output[2]);
            }

大部分在注释中描述的相对明确,但要注意这些操作

复制代码
center.uv = (input[0].uv + input[1].uv + input[2].uv)/3.0;//  计算中间点的uv
center.normalWS = UnityObjectToWorldNormal(centerNormalOS );//  计算中间点的法线
//中心点后期所需数据
center.posWS = mul(unity_ObjectToWorld, centerPosOS);
center.posCS = UnityObjectToClipPos(centerPosOS);
center.dis = noise;

根据一定的计算规则,给中心点赋予详细的顶点信息,这样他就和模型上初始存在的顶点无异,数据填充完毕后和原顶点一起组成片元,组成一个片元的操作如下。

triStream.RestartStrip(); // 重新开始一个新的三角形

triStream.Append(output[1]);

triStream.Append(center);

triStream.Append(output[0]);

通过这样的操作确立一个三角形片元中的三个点。

而这些点在像素着色器中调用时,因为是对每个像素调用,所以会根据这些点取插值,

这里的像素着色器是这么写的

复制代码
float4 frag (g2f i) : SV_Target
{
    return i.dis;
}

所以对于一个片元中的像素,根据片元三个顶点的信息

复制代码
struct g2f
{
    float4 posCS    : SV_POSITION;
    float2 uv       : TEXCOORD0;
    float3 normalWS : TEXCOORD1;
    float3 posWS    : TEXCOORD2;
    float  dis      : TEXCOORD3;
};

插值到对应像素,dis也就会呈现出一种渐变感。

相关推荐
明明明h3 小时前
Unity Assembly Definition & Assembly Definition Reference
unity·游戏引擎
龙中舞王3 小时前
Unity学习笔记(4):人物和基本组件
笔记·学习·unity
无敌最俊朗@6 小时前
unity3d————协程原理讲解
开发语言·学习·unity·c#·游戏引擎
夜色。7 小时前
Unity6 + Android Studio 开发环境搭建【备忘】
android·unity·android studio
这不比博人传燃?12 小时前
传奇996_19——常用函数
游戏引擎
erxij12 小时前
【游戏引擎之路】登神长阶(十四)——OpenGL教程:士别三日,当刮目相看
c++·经验分享·游戏·3d·游戏引擎
咩咩觉主14 小时前
尽量通俗易懂地概述.Net && U nity跨语言/跨平台相关知识
unity·c#·.net·.netcore
墨笺染尘缘15 小时前
Unity——对RectTransform进行操作
ui·unity·c#·游戏引擎
AgilityBaby15 小时前
FairyGUI和Unity联动(入门篇)
unity·游戏引擎
这不比博人传燃?18 小时前
传奇996_19——龙岭总结
游戏引擎