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

相关文章

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

从基础到进阶详解Pandas时间数据处理指南

《从基础到进阶详解Pandas时间数据处理指南》Pandas构建了完整的时间数据处理生态,核心由四个基础类构成,Timestamp,DatetimeIndex,Period和Timedelta,下面我... 目录1. 时间数据类型与基础操作1.1 核心时间对象体系1.2 时间数据生成技巧2. 时间索引与数据

安装centos8设置基础软件仓库时出错的解决方案

《安装centos8设置基础软件仓库时出错的解决方案》:本文主要介绍安装centos8设置基础软件仓库时出错的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录安装Centos8设置基础软件仓库时出错版本 8版本 8.2.200android4版本 javas

Linux基础命令@grep、wc、管道符的使用详解

《Linux基础命令@grep、wc、管道符的使用详解》:本文主要介绍Linux基础命令@grep、wc、管道符的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录grep概念语法作用演示一演示二演示三,带选项 -nwc概念语法作用wc,不带选项-c,统计字节数-

python操作redis基础

《python操作redis基础》Redis(RemoteDictionaryServer)是一个开源的、基于内存的键值对(Key-Value)存储系统,它通常用作数据库、缓存和消息代理,这篇文章... 目录1. Redis 简介2. 前提条件3. 安装 python Redis 客户端库4. 连接到 Re

SpringBoot基础框架详解

《SpringBoot基础框架详解》SpringBoot开发目的是为了简化Spring应用的创建、运行、调试和部署等,使用SpringBoot可以不用或者只需要很少的Spring配置就可以让企业项目快... 目录SpringBoot基础 – 框架介绍1.SpringBoot介绍1.1 概述1.2 核心功能2

Spring Boot集成SLF4j从基础到高级实践(最新推荐)

《SpringBoot集成SLF4j从基础到高级实践(最新推荐)》SLF4j(SimpleLoggingFacadeforJava)是一个日志门面(Facade),不是具体的日志实现,这篇文章主要介... 目录一、日志框架概述与SLF4j简介1.1 为什么需要日志框架1.2 主流日志框架对比1.3 SLF4

Spring Boot集成Logback终极指南之从基础到高级配置实战指南

《SpringBoot集成Logback终极指南之从基础到高级配置实战指南》Logback是一个可靠、通用且快速的Java日志框架,作为Log4j的继承者,由Log4j创始人设计,:本文主要介绍... 目录一、Logback简介与Spring Boot集成基础1.1 Logback是什么?1.2 Sprin

MySQL复合查询从基础到多表关联与高级技巧全解析

《MySQL复合查询从基础到多表关联与高级技巧全解析》本文主要讲解了在MySQL中的复合查询,下面是关于本文章所需要数据的建表语句,感兴趣的朋友跟随小编一起看看吧... 目录前言:1.基本查询回顾:1.1.查询工资高于500或岗位为MANAGER的雇员,同时还要满足他们的姓名首字母为大写的J1.2.按照部门

Android Mainline基础简介

《AndroidMainline基础简介》AndroidMainline是通过模块化更新Android核心组件的框架,可能提高安全性,本文给大家介绍AndroidMainline基础简介,感兴趣的朋... 目录关键要点什么是 android Mainline?Android Mainline 的工作原理关键