Vertex中实现每顶点光照的镜面高光效果
1,基础知识讲解
一个物体在自然界会收到三种光的影响,周围的环境光、漫反射和镜面反射。那么对于计算机要想模拟现实中的光照,就应该也会实现这三种基本光照->环境光、漫反射、镜面高光。对于这三种光照,都存在着一定的影响因素,这就需要去追究其光照模型的公式了。
(1),环境光强影响因素
,由此可看出环境光照的效果取决于一般环境光的强度
和漫反射的材质颜色 
(2),漫反射光强影响因素


由此可看出,漫反射的反射光取决于入射光强度
和材质常数
,对于黑色表面,材质常数
的值是 0,对于白色表面,材质常数
的值是 1。当入射光的方向和物体表面法线的夹角有关,夹角越小,越接近法线,那么接受到的光强也就越多,就越亮。此时的dot乘积也就越大。(回想Cos的函数)
(3),镜面反射光强影响因素


镜面反射通过观察者方向 V 来计算镜面反射,如果 V 接近于 R,反射的强度也会变大(同时也受光泽度
影响),根据 R 与 V 角度余弦值的
-th 次方(pow 值)来生成不同光泽的亮点,我们需要限制负余弦的值为 0,此外镜面反射还要求一个材质颜色
(通常是白色)
2,代码演练
Shader "JiKi/VertexLighting_Specular"
{
Properties
{
_Color("Base RGB",Color)=(,,,)
_Shininess("Shininess",Range(0.001,))=
_SpecColor("Specular Color",Color)=(,,,)
}
SubShader
{
Pass
{
Tags
{
"LightMode"="ForwardBase"
} CGPROGRAM
#pragma vertex vert
#pragma fragment frag
#include "UnityCG.cginc" uniform float4 _LightColor0; uniform float4 _Color;
uniform float _Shininess;
uniform float4 _SpecColor; struct inputVertex
{
float4 vertex:POSITION;
float3 normal:NORMAL;
};
struct outPutVertex
{
float4 pos:SV_POSITION;
float4 col:COLOR;
}; outPutVertex vert(inputVertex input)
{
outPutVertex o;
//工具函数
float4x4 modelMatrix=_Object2World;
float4x4 modelMatrixVerse=_World2Object;
//世界空间中的法线向量
float3 normalDir=normalize(mul(float4(input.normal,0.0),modelMatrixVerse).xyz);
//视线方向
float3 viewDir=normalize(_WorldSpaceCameraPos-mul(modelMatrix,input.vertex).xyz);
//灯光
float3 lightDir;
float attenuation;//衰减量 if(_WorldSpaceLightPos0.w==)//证明是无限长度的方向向量
{
attenuation=1.0;
lightDir=normalize(_WorldSpaceLightPos0).xyz;
}
else
{
float3 vertexToLightSource=(_WorldSpaceLightPos0-mul(modelMatrix,input.vertex)).xyz;
float dist=length(vertexToLightSource);
lightDir=normalize(vertexToLightSource);
attenuation=/dist;
}
//环境光
float3 ambientColor=UNITY_LIGHTMODEL_AMBIENT.rgb*_Color.rgb;
//漫反射
float3 diffuseColor=attenuation*_LightColor0.rgb*_Color.rgb*max(,dot(normalDir,lightDir));
//镜面高光
float3 specularColor;
if(dot(lightDir,normalDir)<)//证明是背面
{
specularColor=float3(,,);
}
else
{
specularColor=attenuation*_LightColor0.rgb*_SpecColor.rgb*pow((max(,dot(viewDir,reflect(-lightDir,normalDir)))),_Shininess);
}
//混合
o.col=float4(ambientColor+diffuseColor+specularColor,1.0);
o.pos=mul(UNITY_MATRIX_MVP,input.vertex);
return o;
} float4 frag(outPutVertex input):COLOR
{
return input.col;
}
ENDCG
}
}
}
最后上效果图
上面代码写的有些简陋,但个人觉得反映出来光照的原理了,有问题还请大家多多指正。
由于这个是基于顶点的,因此光照效果显得有些粗糙,不够细腻,后续将在fragment中去实现光照,那样就会得到光滑的镜面高光。敬请期待!!不早了该睡了哈哈
Vertex中实现每顶点光照的镜面高光效果的更多相关文章
- GLSL逐顶点光照[转]
转载:http://blog.csdn.net/hgl868/article/details/7872350 引言 在OpenGL中有三种类型的光:方向光(directional).点光(point) ...
- Vertex Lit 顶点光照
http://blog.csdn.net/heyuchang666/article/details/51565102 顶点光照(Vertex Lit) 是最低保真度的光照.不支持实时阴影的渲染路径.最 ...
- [Unity Shader] 逐顶点光照和逐片元漫反射光照
书中的6.4节讲的是漫反射的逐顶点光照和逐片元光照. 前一种算法是根据漫反射公式计算顶点颜色(顶点着色器),对颜色插值(光栅化过程)返回每个像素的颜色值(片元着色器). 第二种算法是获得顶点的法线(顶 ...
- unity shader入门(二)语义,结构体,逐顶点光照
下为一个逐顶点漫反射光照shader Shader "study/Chapter6/vertexShader"{ Properties{_Diffuse("Diffuse ...
- Unity可编程管线的顶点光照Shader
UnityCG.cginc有一个叫ShadeVertexLightsFull的函数可以用来计算顶点光照. 源码如下: // Used in Vertex pass: Calculates diffus ...
- Shader Model 3.0:Using Vertex Textures SM3:使用顶点纹理 (NVIDIA spec, 6800支持使用D3DFMT_R32F and D3DFMT_A32B32G32R32F的纹理格式实现Vertex Texture。)
翻译者 周波 zhoubo22@hotmail.com 版权所有 Philipp Gerasimov Randima (Randy) Fernando Simon Green NVIDIA Corpo ...
- ASP.NET中使用JavaScript实现图片自动水平滚动效果
参照网上的资料,在ASP.NET中使用JavaScript实现图片自动水平滚动效果. 1.页面前台代码: <%@ Page Language="C#" AutoEventWi ...
- vue中使用第三方插件animate.css实现动画效果
vue中使用第三方插件animate.css实现动画效果1.首先先引入第三方类animated.css2.将你所需要动画的标签用包裹起来3.在transition元素中添加enter-active-c ...
- MySQL中的模糊查询 like 和 Oracle中的 instr() 函数有同样的查询效果
注:MySQL中的模糊查询 like 和 Oracle中的 instr() 函数有同样的查询效果: 如下所示: MySQL: select * from tableName where name li ...
随机推荐
- SPOJ #442 Searching the Graph
Just CS rookie practice on DFS\BFS. But details should be taken care of: 1. Ruby implementation got ...
- 剑指offer系列23---字符串排列(不是很理解)
[题目]输入一个字符串,按字典序打印出该字符串中字符的所有排列. 例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba. 结果请按字母顺 ...
- (转)VS.NET2010水晶报表安装部署[VS2010]
本文转载自:http://www.cnblogs.com/xiaofengfeng/p/3325793.html 欢迎C#高手加盟QQ群:9340166 水晶报表VS2010版IDE安装标准版SAP ...
- 8张图带你深入理解Java
1.字符串的不变性 下图展示了如下的代码运行过程: String s = "abcd";s = s.concat("ef"); 备注:String refe ...
- android学习笔记40——国际化和资源自适应
国际化——Internationalization,简称I18N. 本地化——Localization,检查L10N. java国际化资源的思路: java提供国际化资源的思路,是将程序中的标签.提示 ...
- [platform]linux platform device/driver(一)--Driver是如何找到对应的device
1.platform device是怎么"自动"关联到platform driver上的? 转向linux driver有些时间了,前段时间碰到个问题,在Linux kernel ...
- windows7修改双系统启动项名称、先后顺序、等待时间
一.进入BCDEdit.exe 正常启动Windows 7 系统,点击“开始” -> “所有程序” -> “附件”,右击“命令提示符” -> “以管理员身份运行”(需要将操作当前用 ...
- c++builder6.0 mdi窗体+自定义子窗体
- NSString类的相关用法
一.NSString字符串连接NSString* string; // 结果字符串 NSString* string1, string2; //已存在的字符串 1. string = [NSStrin ...
- PLSQL_基础系列08_操作符标LPAD / TRUNC / DECODE / TRIM / INSTR(案例)
2014-12-09 Created By BaoXinjian