上一篇文章的例子中我们可以看到顶点着色器的输出参数可以说是直接作为了片段着色器的形参传递过来,那么不由得一个问题浮现出来,顶点着色器的形参是从何处传递过来的?

顶点着色器的形参是gameObject 的meshRenderer组件将所有的mesh数据按每一帧一次传递给OpenGL。

这中间的过程常常被称作一次draw call,往往一次性传输大量mesh信息作为一次draw call 比多次传输少量mesh信息引起多次draw call更加效率。
 
而在上一个例子中我们只接受了MeshRenderer传递过来的 POSITION信息,实际上还有很多信息没有接受。想要在顶点着色器中接收Mesh信息的某些数据,只需要在形参后生命正确的语义即可获得对应的信息。
那么顶点着色器可以根据语义获取到的全部mesh信息有:
float4 vertex : POSITION; //顶点坐标 
float4 tangent : TANGENT; // tangent,三角函数的一种,缩写为tan我们很熟悉了,他的值是mesh到表面法线的正切值
float3 normal : NORMAL; //表面法向量,以对象的坐标系标准化至单位长度 
float4 texcoord : TEXCOORD0;//纹理坐标系的第0个集合 
float4 texcoord1 : TEXCOORD1; //纹理坐标系的第1个集合 
fixed4 color : COLOR;//颜色,通常为常数 
 
同理我们可以声明一个顶点着色器的输入结构体,包含以上所有信息,然后将这个结构体作为形参传递给顶点着色器的入口函数。
 
 

Unity内建的预定义输入结构体:

只要引用UnityCg.cginc头文件(目录Unity > Editor > Data > CGIncludes下)就可以使用预先设定好的结构体直接使用,他们分别有appdata_base  appdata_tan和appdata_full:
struct appdata_base {
float4 vertex : POSITION;
float3 normal : NORMAL;
float4 texcoord : TEXCOORD0;
};
struct appdata_tan { float4 vertex : POSITION;
float4 tangent : TANGENT;
float3 normal : NORMAL;
float4 texcoord : TEXCOORD0;
};
struct appdata_full {
float4 vertex : POSITION;
float4 tangent : TANGENT;
float3 normal : NORMAL;
float4 texcoord : TEXCOORD0;
float4 texcoord1 : TEXCOORD1;
fixed4 color : COLOR; };

可以直接根据需要选择合适的输入结构体,我们可以书写这样形式的代码:

Pass{CGPROGRAM
#pragma vertex vert
#pragma fragment frag
#include "UnityCG.cginc"
struct vertexOutput {
float4 pos : SV_POSITION;
float4 col : TEXCOORD0;
};
vertexOutput vert(appdata_full input)
{
vertexOutput output;
output.pos = mul(UNITY_MATRIX_MVP, input.vertex);
output.col = input.texcoord;
return output;
}
float4 frag(vertexOutput input) : COLOR
{
return input.col;
}
ENDCG}

可以看到运行在球体上的效果:

 

例2:假彩色图像

 
我们了解假彩色图像之前更重要的是把颜色真正的当做一个向量,那么我们专注于这个向量的其中一个分量,令另外的分量固定不变。
例如当我们以球体(sphere)的输入信息中的texcoord参数以语义TEXCOORD0传给片段颜色时,颜色在红色方向的分量是最终texcoord的x坐标呈现的颜色。
意思就是说不管颜色是纯红,纯黄,或者纯洋红色,红色分量始终是1。(计算机的三原色是红绿蓝,黄色并不是三原色之一,黄色也由红色合成)
相对的不管颜色是纯蓝,纯绿,或者纯青色,红色分量始终是0。
那么我们看一下下面这行代码的作用:
output.col = float4(input.texcoord.x, 0.0, 0.0, 1.0);

显然我们取纹理坐标的x参数作为红色分量构造了一个颜色向量,绿色与蓝色分量恒定为0,不透明度恒定为1,那么在球体中出来的应该是一个以纹理坐标的x坐标为变元的一元线性颜色球体。(也可以理解为以地球的纬度为变元,构成了一个由黑色(0,0,0,1)至红色(1,0,0,1)构成的地球)

 
上图为水平反向旋转后看向这个假彩色球体的效果
 
当我们设想我们只关注颜色的红色分量的时候,我们可以看到转动球体从0°旋转至360°再回到0°的时候,红色分量由0变为1,周而复始。它非常类似于行星表面的纬度坐标。
 
同理下面一行代码的作用是以球面的经度为变元构成的黑色至绿色颜色渐变的线性颜色球体:
output.col = float4(0.0, input.texcoord.y, 0.0, 1.0);

可以想象这个球体的南极应该是黑色(0,0,0,1),北极应该是绿色(0,1,0,1)

 
上图为从球体的南极看向这个假彩色球体的效果
 
写到这里,其实假彩色(False Color)这个概念玩过PS的朋友应该可以马上联想到通道,其实就是红色通道 绿色通道和蓝色通道而已,只不过我们以向量的观点来看他们是颜色向量的分量而已。我们可以理解为颜色为3个通道的混合,也可以认为是3个分量的合向量
 
 
 
纹理坐标(texture coordinates)非常适合用来表现颜色,因为他们的值范围都是0~1,所以无需任何转换就能把一个纹理坐标与一个颜色一一映射。
坐标上的法向量同样可以用来表现颜色,但是需要一定的转换。因为法向量的值范围是-1至+1,假设一个坐标x,y,z的法向量是α,那么α的范围是(-1,-1,-1)至(1,1,1),非常轻松的数学转换就可以将他与一个颜色一一映射:
 
α与(1,1,1)进行矢量相加,然后与2相除,即可得到一个(0,0,0)至(1,1,1)的向量,再补充第四元不透明度即可称为一个RGBA颜色向量。所以代码为:
output.col = float4((input.normal + float3(1.0, 1.0, 1.0)) / 2.0, 1.0);

这里我不再把效果图给出来了,大家可以自己试试

由此可见,我们如果想让mesh信息中的数据以颜色体现出来,如果原信息的值域的每个分量不在[0,1] 只需要将原信息的值域变换至这个区间内,使其每个分量都不超过[0,1]这个区间,我们就能将mesh信息与颜色建立单向映射体现出来。

 
 
那么下面3行代码中返回的颜色是否是正确的颜色,只用判断col向量的分量是否超过了[0,1]区间,超出这个区间我们就只能看到黑色:
output.col = input.texcoord - float4(1.5, 2.3, 1.1, 0.0);  //rgba区间分别为[-1.5,0.5],[-2.3,-1.3],[-1.1,-0.1],[0,1],不能与颜色建立映射
output.col = float4(input.texcoord.z);//rgba区间分别为[0,1],[0,1],[0,1],[0,1],可以与颜色建立映射
output.col = input.texcoord / tan(0.0);//分母为0数学上无意义,因为值无穷大,所以区间是[0,∞]

再看复杂一点的,判断下面的输出颜色是否正确,需要一些点乘和叉乘的知识:

output.col = dot(input.normal, float3(input.tangent)) *input.texcoord;
output.col = dot(cross(input.normal, float3(input.tangent)),input.normal) * input.texcoord;
output.col = float4(cross(input.normal, input.normal), 1.0);
output.col = float4(cross(input.normal,float3(input.vertex)), 1.0);
 
不好意思我的点乘和叉乘已经全部还给老师了,看来是得把我送回学校去重新学学了!我解不出来~

弧度函数radians()与噪音函数noise()总是返回黑色:

 
output.col = radians(input.texcoord);
output.col = noise(input.texcoord);

[转]解读Unity中的CG编写Shader系列二的更多相关文章

  1. 解读Unity中的CG编写Shader系列二

    转自 http://www.itnose.net/detail/6095974.html 上一篇文章的例子中我们可以看到顶点着色器的输出参数可以说是直接作为了片段着色器的形参传递过来,那么不由得一个问 ...

  2. 解读Unity中的CG编写Shader系列八(镜面反射)

    转自http://www.itnose.net/detail/6117378.html 讨论完漫反射之后,接下来肯定就是镜面反射了 在开始镜面反射shader的coding之前,要扩充一下前面提到的知 ...

  3. [转]解读Unity中的CG编写Shader系列9——镜面反射

    讨论完漫反射之后,接下来肯定就是镜面反射了在开始镜面反射shader的coding之前,要扩充一下前面提到的知识,加深理解镜面反射与漫反射的区别.注:这篇文章实现的镜面反射是逐顶点着色(per-ver ...

  4. [转]解读Unity中的CG编写Shader系列7——漫反射

    如果前面几个系列文章的内容过于冗长缺乏趣味着实见谅,由于时间原因前面的混合部分还没有写完,等以后再补充,现在开始关于反射的内容了.折射与反射在物理世界中,光的反射与折射往往是同时存在的,光源由真空或者 ...

  5. 解读Unity中的CG编写Shader系列七(不透明度与混合)

    转自http://www.itnose.net/detail/6098539.html 1.不透明度 当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段 ...

  6. 解读Unity中的CG编写Shader系列三

    转自http://www.itnose.net/detail/6096068.html 在上一个例子中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上.这篇文章 ...

  7. [转]解读Unity中的CG编写Shader系列6——不透明度与混合

    1.不透明度当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到 ...

  8. [转]解读Unity中的CG编写Shader系列3——表面剔除与剪裁模式

    在上一个例子中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上.这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后面 ...

  9. 解读Unity中的CG编写Shader系列3——表面剔除与剪裁模式

    在上一个样例中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上. 这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后 ...

随机推荐

  1. dubbo学习 一 dubbo概述

    1,背景     1,网站刚开时候的时候可能所有的功能业务都在一个应用里面 2,当业务不断复杂,流量不断增多的时候,就需要将原先的一个应用划分成多个独立的应用. 3,当分出来的业务越来越多的时候,应用 ...

  2. .net 连接ORACLE中文显示乱码解决方案

    FYI由于历史的原因,早期的oracle没有中文字符集(如oracle6.oracle7.oracle7.1),但有的用户从那时起就使用数据库了, 并用US7ASCII字符集存储了中文,或是有的用户在 ...

  3. 上传图片用图片文件的对象hash哈希值判断图片是否一样,避免重复提交相同的图片到服务器中

    上传图片用图片文件的对象hash哈希值判断图片是否一样,避免重复提交相同的图片到服务器中 前端:要用到一个插件,点击下载 <!DOCTYPE html> <html xmlns=&q ...

  4. 动态规划——最长不下降子序列(LIS)

    最长不降子序列是这样一个问题: 下面介绍动态规划的做法. 令 dp[i] 表示以 A[i] 结尾的最长不下降序列长度.这样对 A[i] 来说就会有两种可能: 如果存在 A[i] 之前的元素 A[j] ...

  5. 服务器发送邮件出现Could not connect to SMTP host错误 解决办法

    服务器发送邮件出现Could not connect to SMTP host错误 解决办法 功夫不负有心人,最后了解到,除了google的smtp服务器收到请求“smtp”会接受,其他服务器比如qq ...

  6. 使用jq.lazyload.js,解决设置loading图片的问题

    最近在使用lazyload的时候,遇上一个问题.当对img做宽100%时,就是placeholder的loading图片也会100%宽,这样一般来说loading图片就会变得很大.实在是不能应用到项目 ...

  7. 用Golang为Python编写模块

    Go里面需要显示的引入C模块, 让编译器支持生成动态链接库, 并且在代码中可以使用C语言的数据类型,这个至关重要. Calling Go code from Python code 摘取一个最简单例子 ...

  8. Spring总结二:IOC(控制反转)xml方式

    1,简介: IoC :Inverse of control 控制反转 ,思想就是在项目中引入一个工厂容器,对项目中接口依赖对象的创建,实现项目中对于依赖对象解耦合. 将程序中对象的创建权以及对象的整个 ...

  9. Django中间件CsrfViewMiddleware源码分析

    Django Documentation csrf保护基于以下: 1, 一个CSRF cookie基于一个随机生成的值,其他网站无法得到,次cookie有CsrfViewMiddleware产生.它与 ...

  10. LoadRunner 关联和集合点、检查点

    1)关联的定义 很多时候,当时录完之后,没有问题.过一段时间再跑脚本,就不会成功.比如session,过期了,再一次使用,就会出错.这个时候,需要在每次访问的时候动态的拿到session,这种情况就需 ...