解读Unity中的CG编写Shader系列1——初识CG

CG=C for Graphics  用于计算机图形编程的C语言超集

前提知识点:

1.CG代码必须用

CGPROGRAM

。。。

ENDCG括起来

2.顶点着色器与片段着色器的主函数名称可任意,但须要再#pragma vert 与#pragma fragment中声明而且与主函数名全然匹配,shader才会找到入口

3.float4是一种压缩数组,float4 vert与float vert[4]严格意义上讲不同,尽管都是存放4个float,但float4作为向量类型做点乘、内积等处理更高速

4.语义 :变量除了变量名与数据类型之外,还在:后声明其语义

样例一:RGB立方体

例题阐述:

给一个立方体(cube)创建一个shader,在立方体的直角坐标系中,三个坐标x,y,z的取值范围为{0,0,0}至{1,1,1},而RGBA颜色red,green,blue,alpha的取值范围正好是{0,0,0,0}至{1,1,1,1},将alpha写死为1,那么r,g,b分别与x,y,z建立一一映射,可得到一个6个面展示全部RGB颜色的立方体如图:

详细代码及其含义例如以下:

Shader "Custom/RGBCube" {
	SubShader
	{
		Pass {
		CGPROGRAM
		#pragma vertex vert //顶点着色器入口函数声明
		#pragma fragment frag // 片段着色器入口函数声明
		//顶点输出结构体
		struct vertexOutput {
			//声明结构体的成员pos,类型为float类型的4元向量,语义为SV_POSITION,col同理;
			float4 pos : SV_POSITION;
			float4 col : TEXCOORD0;
		};
		//顶点着色器入口函数vert,与pragma第一条声明匹配,返回类型为刚刚定义的顶点输出结构体
		vertexOutput vert(float4 vertexPos : POSITION)

		{
			vertexOutput output; //这里不须要structkeyword
			//顶点着色器将数据写入输出结构体中。
			output.pos = mul(UNITY_MATRIX_MVP, vertexPos);
			//mul是顶点变换函数,UNITY_MATRIX_MVP是unity的内建矩阵,vertexPos是这个函数的形參
			//此行代码的作用为将形參vertexPos(本例即Cube对象的顶点向量)依照unity的内建矩阵进行顶点变换
			output.col = vertexPos + float4(0.5, 0.5, 0.5, 0.0);
			//这行代码是实现RGB立方体的关键
			//vertexPos的值域为题干所提到的x,y,z三元组各自减去0.5构成的值域
			//可是这里接受的类型为float4,可见第四元应该是无意义的常数1
			//意思是vertexPos的值域为{-0.5,-0.5,-0.5,1}至{0.5,0.5,0.5,1}
			//而对这个值域进行+{0.5,0.5,0.5,0}的矢量相加才干得到RGB (A恒定为1)的全部颜色区间

			return output;
			//将输出结构体返回,进入下一个环节(简单理解为给片段着色器)
			//ps:更仔细的环节有顶点变换-->顶点着色-->几何元的构建-->光栅化几何元
			//-->片段着色-->略
		}
		//片段着色器入口函数frag,与pragma第二条声明匹配,返回类型为float4语义为COLOR,
		//这里除了颜色没有其它的输出,所以没有输出结构体
		float4 frag(vertexOutput input) : COLOR
		//此函数的形參类型为顶点着色器的输出结构体,没有语义
		//原因就在于片段着色器位于顶点着色器的下一个环节,參数依照这个顺序传递
		{
			//因为col属性已经在顶点着色器中计算,直接返回进入下一环节
			//下一环节是什么这里不探讨了
			return input.col;
		}
		ENDCG
		}
	}
	//假设以上SubShader渲染失败则回滚採用Diffuse
	FallBack "Diffuse"
}

补充:

刚刚写掉了,为何vertexPos的值域为{-0,5,-0.5,-0.5,1}至{0.5,0.5,0.5,1}而非{0,0,0,1}至{1,1,1,1}呢?

由于我们的直角坐标系原点没有在顶点上而是在cube的几何中心,故其值域为{-0,5,-0.5,-0.5,1}至{0.5,0.5,0.5,1},over

假设我的博客对您有帮助或者您有不论什么疑问,欢迎增加重庆u3d交流QQ群我会给您解答:68994667,也能够加群与我们一同交流技术

时间: 2024-11-05 12:37:40

解读Unity中的CG编写Shader系列1——初识CG的相关文章

解读Unity中的CG编写Shader系列一

CG=C for Graphics  用于计算机图形编程的C语言超集 前提知识点: 1.CG代码必须用 CGPROGRAM ... ENDCG括起来 2.顶点着色器与片段着色器的主函数名称可随意,但需要再#pragma vert 与#pragma fragment中声明并且与主函数名完全匹配,shader才会找到入口 3.float4是一种压缩数组,float4 vert与float vert[4]严格意义上讲不同,虽然都是存放4个float,但float4作为向量类型做点乘.内积等处理更快速

解读Unity中的CG编写Shader系列四——unity中的圆角矩形shader

上篇文章中我们掌握了表面剔除和剪裁模式 这篇文章将利用这些知识实现一个简单的,但是又很常用的例子:把一张图片做成圆角矩形 例3:圆角矩形Shader 好吧我承认在做这个例子的时候走了不少弯路,由于本人对矩阵的知识掌握已经悉数还给老师,所以一开始用了一些笨办法计算圆角矩形区域. 我们知道TEXTCOORD0是一个以对象为坐标系的坐标,并且范围在该坐标的第一象限,取值为(0,0)到(1,1) 那么我们把每一张图片都看做一张1X1大小的矩形 我们要在1X1大小的矩形中擦除4个角,应该是这样: 以左上角

解读Unity中的CG编写Shader系列9——镜面反射

讨论完漫反射之后,接下来肯定就是镜面反射了 在开始镜面反射shader的coding之前,要扩充一下前面提到的知识,加深理解镜面反射与漫反射的区别. 引用一下一位前人博文中的一些基础概念,特别是关于冯氏反射模型的: 平行光(directional light) 一种是从特定方向射入并只会照亮面对入射方向的物体,我们称之为平行光(directional light). 环境光(ambient light) 另一种光是来自所有方向并且会照亮所有物体,不管这些物体的朝向如何,我们称之为环境光(ambi

解读Unity中的CG编写Shader系列6——漫反射

如果前面几个系列文章的内容过于冗长缺乏趣味着实见谅,由于时间原因前面的混合部分还没有写完,等以后再补充,现在开始关于反射的内容了. 折射与反射 在物理世界中,光的反射与折射往往是同时存在的,光源由真空或者空气中射入一种材料,光在进入这种材料的同时就发生了折射,折射的程度与各个介质的折射率有关,使光的传播路线偏离原来的路线: 继而如果光在通过不同传播介质的表面时,会像乒乓球一样弹回来,我们人眼能够看到东西,都是因为东西会反射光源,如果一种物质无法反射光,或者没有光源,我们就看不到东西.同样对于不同

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

1.不透明度 当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到底以怎样的形式混合在一起最后输出到帧缓存中是我们现在首要讨论的内容. 1.混合(blending) 上一篇文章中的管道环节中的"逐帧操作"环节中的一项操作就是混合操作,可见混合操作是不可编程的固定功能环节. 在混合操作中,我们将片段着色器中计算出来的颜色称之为 "源颜色&quo

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

在上一个样例中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上. 这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后面剪裁(front face culling and back face culling)来达到透明效果. 当一个mesh组件的信息被传递后,我们能够通过代码决定哪些部分渲染(render)出来.而哪些部分不要.这个过程就像把那些不要的部分剔除了,我们看不到他.尽管他的mesh信息还在.可是我们的G

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

转自http://www.itnose.net/detail/6096068.html 在上一个例子中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上.这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后面剪裁(front face culling and back face culling)来达到透明效果. 当一个mesh组件的信息被传递后,我们可以通过代码决定哪些部分渲染(render)出来,而哪些部分不要,这

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

转自http://www.itnose.net/detail/6098539.html 1.不透明度 当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到底以怎样的形式混合在一起最后输出到帧缓存中是我们现在首要讨论的内容. 1.混合(blending) 上一篇文章中的管道环节中的“逐帧操作”环节中的一项操作就是混合操作,可见混合操作是不可编程的固定功能环节. 在

Unity中的CG编写Shader系列(Blend)

1.不透明度 当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到底以怎样的形式混合在一起最后输出到帧缓存中是我们现在首要讨论的内容. 1.混合(blending) 上一篇文章中的管道环节中的“逐帧操作”环节中的一项操作就是混合操作,可见混合操作是不可编程的固定功能环节. 在混合操作中,我们将片段着色器中计算出来的颜色称之为 “源颜色”,帧缓存中对应的像素已经存