[基础] Unity Shader:顶点着色器(vert)函数

2024-05-03 06:20

本文主要是介绍[基础] Unity Shader:顶点着色器(vert)函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

顶点着色器(Vertex Shader)是图形渲染的第一个阶段,它的输入来自于CPU。顶点着色器的处理单位是顶点,CPU输入进来的每个顶点都会调用一次顶点着色器函数,也就是我们在Shader代码里所定义的vert函数。本篇我们将会通过顶点的颜色变换、顶点位移、顶点扭曲来介绍顶点着色器(vert)函数。

1. 基于模型空间的顶点颜色变换

3D模型中的每个顶点都有一个处于模型空间里的顶点坐标,顶点坐标通常在经过顶点着色器处理后会被转换到规范化设备坐标(NDC)空间,这个空间中的x、y、z坐标范围通常是[-1, 1],在vert函数中,我们就可以利用这些坐标来对顶点的颜色进行变换。我们在场景里创建一个立方体,它的坐标值一般为0.5或-0.5,例如,我们要将坐标为(0.5, 0.5, 0.5)的顶点渲染成红色,其它顶点为白色,则Cg代码如下:

if (appData.vertex.x == 0.5 && appData.vertex.y == 0.5 && appData.vertex.z == 0.5)outData.color = fixed4(1, 0, 0, 1);
elseoutData.color = fixed4(1, 1, 1, 1);

实现效果如下图所示:

我们发现,靠近立方体顶点(0.5, 0.5, 0.5)的部分呈现出红色,且不管我们如何移动或旋转立方体,该顶点区域依然保持为红色,这是因为我们使用的是基于模型空间的固定顶点坐标。

2. 基于世界空间的顶点颜色变换

倘若我们要让模型的顶点接近世界空间的某个位置时才让颜色发生改变,我们则需要将模型顶点变为世界顶点。要这么做,我们先要让顶点进行物体到世界的矩阵变换:

// 将顶点进行物体到世界的矩阵变换
float4 wpos = mul(unity_ObjectToWorld, appData.vertex);

接下来,例如我们要实现当模型顶点大于世界x坐标 0 时,则变为红色,反之为白色,Cg代码如下:

if (wpos.x > 0)outData.color = fixed4(1, 0, 0, 1);
elseoutData.color = fixed4(1, 1, 1, 1);

实现效果如下图所示:

 3. 顶点位移

当然,我们也可以在vert函数里修改每个顶点的位置,例如,我们让越靠近模型圆心的顶点,它的y值越高:

Properties
{_R("Range", range(0, 5)) = 1
}
float2 xy = appData.vertex.xz;
float d = _R - length(xy); // R - 计算原点到顶点的长度
d = d < 0 ? 0 : d;
float baseHeight = 1;
float4 uppos = float4(appData.vertex.x, baseHeight * d, appData.vertex.z, appData.vertex.w);
outData.pos = UnityObjectToClipPos(uppos);

我们在编辑器中拖动 Range 的值,效果如下:

 4. 顶点扭曲

我们也可以通过让每个顶点通过旋转矩阵变换来实现模型的扭曲,Cg代码如下:

// 旋转角度
float angle = length(appData.vertex) * _SinTime.w;
// 定义旋转矩阵
float4x4 m = {float4(cos(angle), 0, sin(angle), 0),float4(0, 1, 0, 0),float4(-sin(angle), 0, cos(angle), 0),float4(0, 0, 0, 1)
};
// 与MVP矩阵相乘
appData.vertex = mul(m, appData.vertex);
outData.pos = UnityObjectToClipPos(appData.vertex);

效果如下图所示:

这篇关于[基础] Unity Shader:顶点着色器(vert)函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

C#基础之委托详解(Delegate)

《C#基础之委托详解(Delegate)》:本文主要介绍C#基础之委托(Delegate),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 委托定义2. 委托实例化3. 多播委托(Multicast Delegates)4. 委托的用途事件处理回调函数LINQ

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、