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

2024-03-13 08:20

本文主要是介绍Unity URP 如何写基础的几何着色器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

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

下面是完整代码

Shader "Kerzh/GeoShaderTest"
{Properties{_Length("Length",Float) = 1}SubShader{Tags {"LightMode" = "UniversalForward"}LOD 100Pass{Cull OffCGPROGRAM#pragma vertex vert#pragma fragment frag#pragma geometry geom#include "UnityCG.cginc"//cpu 传到 vsstruct appdata{float4 posOS : POSITION;float2 uv : TEXCOORD0;float4 normalOS : NORMAL;};//vs 传到 gsstruct v2g{float4 posOS    : TEXCOORD0;float2 uv       : TEXCOORD1;float3 normalOS : TEXCOORD2;};//gs 传到 fsstruct 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;//  计算中间点的uvfloat3 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;//  计算中间点的uvfloat3 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也就会呈现出一种渐变感。

这篇关于Unity URP 如何写基础的几何着色器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/804206

相关文章

0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型的操作流程

《0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeekR1模型的操作流程》DeepSeekR1模型凭借其强大的自然语言处理能力,在未来具有广阔的应用前景,有望在多个领域发... 目录0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型,3步搞定一个应

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu

uva 1342 欧拉定理(计算几何模板)

题意: 给几个点,把这几个点用直线连起来,求这些直线把平面分成了几个。 解析: 欧拉定理: 顶点数 + 面数 - 边数= 2。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#inc

XTU 1237 计算几何

题面: Magic Triangle Problem Description: Huangriq is a respectful acmer in ACM team of XTU because he brought the best place in regional contest in history of XTU. Huangriq works in a big compa

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

poj 3304 几何

题目大意:给出n条线段两个端点的坐标,问所有线段投影到一条直线上,如果这些所有投影至少相交于一点就输出Yes!,否则输出No!。 解题思路:如果存在这样的直线,过投影相交点(或投影相交区域中的点)作直线的垂线,该垂线(也是直线)必定与每条线段相交,问题转化为问是否存在一条直线和所有线段相交。 若存在一条直线与所有线段相交,此时该直线必定经过这些线段的某两个端点,所以枚举任意两个端点即可。

POJ 2318 几何 POJ 2398

给出0 , 1 , 2 ... n 个盒子, 和m个点, 统计每个盒子里面的点的个数。 const double eps = 1e-10 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;}struct Point{double x , y