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


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

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

    下面是完整代码

    1. Shader "Kerzh/GeoShaderTest"
    2. {
    3. Properties
    4. {
    5. _Length("Length",Float) = 1
    6. }
    7. SubShader
    8. {
    9. Tags {"LightMode" = "UniversalForward"}
    10. LOD 100
    11. Pass
    12. {
    13. Cull Off
    14. CGPROGRAM
    15. #pragma vertex vert
    16. #pragma fragment frag
    17. #pragma geometry geom
    18. #include "UnityCG.cginc"
    19. //cpu 传到 vs
    20. struct appdata
    21. {
    22. float4 posOS : POSITION;
    23. float2 uv : TEXCOORD0;
    24. float4 normalOS : NORMAL;
    25. };
    26. //vs 传到 gs
    27. struct v2g
    28. {
    29. float4 posOS : TEXCOORD0;
    30. float2 uv : TEXCOORD1;
    31. float3 normalOS : TEXCOORD2;
    32. };
    33. //gs 传到 fs
    34. struct g2f
    35. {
    36. float4 posCS : SV_POSITION;
    37. float2 uv : TEXCOORD0;
    38. float3 normalWS : TEXCOORD1;
    39. float3 posWS : TEXCOORD2;
    40. float dis : TEXCOORD3;
    41. };
    42. float _Length;
    43. v2g vert (appdata input)
    44. {
    45. v2g output;
    46. output.uv = input.uv;
    47. output.normalOS = input.normalOS;
    48. output.posOS = input.posOS;
    49. return output;
    50. }
    51. //最大的顶点数,这里比如你要生成三个三角形面片,那么一个面片需要三个顶点,就是9个顶点
    52. [maxvertexcount(9)]
    53. void geom (triangle v2g input[3],inout TriangleStream<g2f> triStream)
    54. {
    55. g2f center; // 中间那个点
    56. center.uv = (input[0].uv + input[1].uv + input[2].uv)/3.0;// 计算中间点的uv
    57. float3 centerNormalOS = (input[0].normalOS + input[1].normalOS + input[2].normalOS)/3.0;
    58. center.normalWS = UnityObjectToWorldNormal(centerNormalOS );// 计算中间点的法线
    59. float3 centerPosOS = (input[0].posOS + input[1].posOS + input[2].posOS)/3.0;// 计算中间点的位置
    60. centerPosOS = centerPosOS + centerNormalOS *_Length;// 沿着中心点法线根据噪声及系数位移中心点
    61. //中心点后期所需数据
    62. center.posWS = mul(unity_ObjectToWorld, centerPosOS);
    63. center.posCS = UnityObjectToClipPos(centerPosOS);
    64. center.dis = 1;
    65. g2f output[3]; // 这里是原样把输入三角形的三个点放到了这里面,只是为了下面取得方便
    66. for(int i=0;i<3;i++)
    67. {
    68. g2f p0;
    69. p0.uv = input[i].uv;
    70. p0.normalWS = UnityObjectToWorldNormal( input[i].normalOS);
    71. p0.posWS = mul(unity_ObjectToWorld, input[i].posOS);
    72. p0.posCS = UnityObjectToClipPos(input[i].posOS );
    73. p0.dis = 0;
    74. output[i] = p0;
    75. }
    76. // 根据这三个点分别和中心点制造三角形输出
    77. triStream.RestartStrip(); // 重新开始一个新的三角形
    78. triStream.Append(output[1]);
    79. triStream.Append(center);
    80. triStream.Append(output[0]);
    81. triStream.RestartStrip(); // 重新开始一个新的三角形
    82. triStream.Append(output[2]);
    83. triStream.Append(center);
    84. triStream.Append(output[1]);
    85. triStream.RestartStrip(); // 重新开始一个新的三角形
    86. triStream.Append(output[0]);
    87. triStream.Append(center);
    88. triStream.Append(output[2]);
    89. }
    90. float4 frag (g2f i) : SV_Target
    91. {
    92. return i.dis;
    93. return float4(i.uv,0,0);
    94. }
    95. ENDCG
    96. }
    97. }
    98. }

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

    1. //最大的顶点数,这里比如你要生成三个三角形面片,那么一个面片需要三个顶点,就是9个顶点
    2. [maxvertexcount(9)]
    3. void geom (triangle v2g input[3],inout TriangleStream<g2f> triStream)
    4. {
    5. g2f center; // 中间那个点
    6. center.uv = (input[0].uv + input[1].uv + input[2].uv)/3.0;// 计算中间点的uv
    7. float3 centerNormalOS = (input[0].normalOS + input[1].normalOS + input[2].normalOS)/3.0;
    8. center.normalWS = UnityObjectToWorldNormal(centerNormalOS );// 计算中间点的法线
    9. float3 centerPosOS = (input[0].posOS + input[1].posOS + input[2].posOS)/3.0;// 计算中间点的位置
    10. centerPosOS = centerPosOS + centerNormalOS *_Length;// 沿着中心点法线根据噪声及系数位移中心点
    11. //中心点后期所需数据
    12. center.posWS = mul(unity_ObjectToWorld, centerPosOS);
    13. center.posCS = UnityObjectToClipPos(centerPosOS);
    14. center.dis = 1;
    15. g2f output[3]; // 这里是原样把输入三角形的三个点放到了这里面,只是为了下面取得方便
    16. for(int i=0;i<3;i++)
    17. {
    18. g2f p0;
    19. p0.uv = input[i].uv;
    20. p0.normalWS = UnityObjectToWorldNormal( input[i].normalOS);
    21. p0.posWS = mul(unity_ObjectToWorld, input[i].posOS);
    22. p0.posCS = UnityObjectToClipPos(input[i].posOS );
    23. p0.dis = 0;
    24. output[i] = p0;
    25. }
    26. // 根据这三个点分别和中心点制造三角形输出
    27. triStream.RestartStrip(); // 重新开始一个新的三角形
    28. triStream.Append(output[1]);
    29. triStream.Append(center);
    30. triStream.Append(output[0]);
    31. triStream.RestartStrip(); // 重新开始一个新的三角形
    32. triStream.Append(output[2]);
    33. triStream.Append(center);
    34. triStream.Append(output[1]);
    35. triStream.RestartStrip(); // 重新开始一个新的三角形
    36. triStream.Append(output[0]);
    37. triStream.Append(center);
    38. triStream.Append(output[2]);
    39. }

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

    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也就会呈现出一种渐变感。

  • 相关阅读:
    企业为什么选择SDWAN代替MPLS?
    微服务架构最佳实践:消息队列
    Llama 2 来袭 - 在 Hugging Face 上玩转它
    【Jenkins打包服务,Dockerfile报错:manifest for java : 8 not fourd】
    测试工程师面试之设计测试用例
    C# 简单封装异步Socket Server
    【代码学习】聚类方法及实现记录
    【毕业设计】基于php+mysql+apache的网络数据包分析工具设计与实现(毕业论文+程序源码)——网络数据包分析工具
    asp毕业设计——基于asp+access的网页设计辅导系统设计与实现(毕业论文+程序源码)——网页设计辅导系统
    用友NC-Cloud uploadChunk 任意文件上传漏洞
  • 原文地址:https://blog.csdn.net/qq_55895529/article/details/136666484