本系列主要参考《Unity Shaders and Effects Cookbook》一书(感谢原书作者),同时会加上一点个人理解或拓展。
这里是本书所有的插图。这里是本书所需的代码和资源(当然你也可以从官网下载)。
========================================== 分割线 ==========================================
啦啦啦~我又回来看书啦。文章开始说一些题外话。因为写博客和在群里比较活跃(为什么QQ给我的称谓是“吐槽”。。。我这么乖。。。),最近有一些朋友给我发邮件或者私信,很多人都给了我鼓励,我很开心,也有人给了我很好的学习建议,这些经历让我觉得坚持写博客是正确的一条路~也谢谢所有鼓励、支持和关注我的人!我会坚持下去的。
好啦,言归正传~这篇开始,我决定开始一个新的章节——Vertex Magic。顾名思义,就是学习顶点相关的内容。自从看了下面Unity Gems里面的一张图,我对整个Surface Shader的工作流程有了更好的理解,在此再重温一下。
可以看出来共有四个阶段我们可以参与。我们之前学习的大多是在上图的第二和第三个阶段,也就是编写surf和LightingXXX函数来影响表面着色和光照函数的。在卡通风格的Shader中,我们也对最后一个更改像素颜色的机会——finalcolor命令有了一个初步的尝试。而现在,我们将用一章的时间来学习如何利用第一个阶段——顶点函数来影响像素颜色。
顶点函数是在每个顶点被传送到GPU之前被调用一次。它的作用是从模型坐标系中得到三维坐标,然后再转换到其渲染到屏幕时在屏幕坐标系中的二维位置。因此,通过顶点函数,我们可以修改顶点的位置、颜色和UV坐标。一旦我们完成了对顶点的修改后,就会进入到surf函数的执行。与顶点函数是逐顶点执行的方式不同,surf函数则是逐像素执行的。
通过顶点函数,我们可以创造像海上的波浪、旗帜飘动的动态效果,或者使用Shader来给顶点着色。这一篇,我们来学习如何在一个Surface Shader中创建一个最简单的顶点函数!
在深入学习顶点函数之前,我们得首先了解如何通过顶点函数得到和存储顶点相关信息。
你的场景应该看起来是这样的:
下面,我们开始编写Shader。
Properties { _MainTint("Global Color Tint", Color) = (1,1,1,1) }
CGPROGRAM #pragma surface surf Lambert vertex:vert
float4 _MainTint;
struct Input { float2 uv_MainTex; float4 vertColor; };
void vert(inout appdata_full v, out Input o) { o.vertColor = v.color; }
void surf (Input IN, inout SurfaceOutput o) { o.Albedo = IN.vertColor.rgb * _MainTint.rgb; }
Shader "Custom/SimpleVertexColor" { Properties { _MainTint("Global Color Tint", Color) = (1,1,1,1) } SubShader { Tags { "RenderType"="Opaque" } LOD 200 CGPROGRAM #pragma surface surf Lambert vertex:vert float4 _MainTint; struct Input { float2 uv_MainTex; float4 vertColor; }; void vert(inout appdata_full v, out Input o) { o.vertColor = v.color; } void surf (Input IN, inout SurfaceOutput o) { o.Albedo = IN.vertColor.rgb * _MainTint.rgb; } ENDCG } FallBack "Diffuse" }
void surf (Input IN, inout SurfaceOutput o) { o.Albedo = IN.vertColor.rgb * _MainTint.rgb; o.Alpha = IN.vertColor.a; }