untiy 3d ShaderLab_第9章_1_平面阴影(一)
来源:互联网 发布:尚学堂java ppt 编辑:程序博客网 时间:2024/05/19 23:12
九.平面阴影
平面阴影是一种比较特殊的情形。在这种情形里,我们只考虑物体的阴影投射到平面上的情形,所以有一套相对比较简单的专用算法。
9.1平行光对平面的投影
9.1.1对平行光投影的考虑
首先考虑最简单的情况,如何计算一个平行光的投影。平行光在我们的计算中其实就是一个方向矢量,是阴影的投射方向,而平面是阴影要影响的目标物体。我们需要知道到目标物体的Object Space矩阵,在目标物体的空间内将投影物体的顶点进行重新计算,计算其沿光线方向,在阴影接受平面上的位置,这个位置关系可以通过三角形相似来计算。如果我们使用Unity自带的Plane作为阴影接受平面,那么我们只需要重新计算顶点的xz位置,如果阴影投射到Build In的Plane上,那么在其Object Space中,y应该为0,但是实际使用时,为了保证阴影永远在物体上面,我们会对z进行偏移。
如图所示,一个精准的阴影己经出现在平面上,先展示效果。
9.1.2进出阴影接受平面的矩阵
首先,每一个投射平面阴影的物体都需要下面这么一个脚本来告诉它阴影接受物体的信息,
9.1.3使用三角形相似计算阴影
平面阴影是一种比较特殊的情形。在这种情形里,我们只考虑物体的阴影投射到平面上的情形,所以有一套相对比较简单的专用算法。
9.1平行光对平面的投影
9.1.1对平行光投影的考虑
首先考虑最简单的情况,如何计算一个平行光的投影。平行光在我们的计算中其实就是一个方向矢量,是阴影的投射方向,而平面是阴影要影响的目标物体。我们需要知道到目标物体的Object Space矩阵,在目标物体的空间内将投影物体的顶点进行重新计算,计算其沿光线方向,在阴影接受平面上的位置,这个位置关系可以通过三角形相似来计算。如果我们使用Unity自带的Plane作为阴影接受平面,那么我们只需要重新计算顶点的xz位置,如果阴影投射到Build In的Plane上,那么在其Object Space中,y应该为0,但是实际使用时,为了保证阴影永远在物体上面,我们会对z进行偏移。
如图所示,一个精准的阴影己经出现在平面上,先展示效果。
9.1.2进出阴影接受平面的矩阵
首先,每一个投射平面阴影的物体都需要下面这么一个脚本来告诉它阴影接受物体的信息,
具体来说就是到其Object Space空间的矩阵,以及从平面的Object Space返回的矩阵。
using UnityEngine;using System.Collections;[ExecuteInEditMode]public class PlaneShadowCaster : MonoBehaviour { public Transform reciever;void Update () { GetComponent<Renderer>().sharedMaterial.SetMatrix("_World2Ground",reciever.GetComponent<Renderer>().worldToLocalMatrix); GetComponent<Renderer>().sharedMaterial.SetMatrix("_Ground2World", reciever.GetComponent<Renderer>().localToWorldMatrix);}}
9.1.3使用三角形相似计算阴影
知道如何出入阴影接受平面的矩阵后,我们就可以对一个投射阴影的物体计算阴影了。这是在PlanarShadow l.shader中完成的,
其完全代码如下:
Shader "Tut/Shadow/PlanarShadow_1" {SubShader {pass { //对物体本身做一个简单的光照计算 Tags { "LightMode" = "ForwardBase" }Material{Diffuse(1,1,1,1)}Lighting On}pass { Tags { "LightMode" = "ForwardBase" } Blend DstColor SrcColorOffset -1,-1//使阴影在平面之上CGPROGRAM#pragma vertex vert #pragma fragment frag#include "UnityCG.cginc"float4x4 _World2Ground;float4x4 _Ground2World;float4 vert(float4 vertex: POSITION) : SV_POSITION{float3 litDir;litDir=WorldSpaceLightDir(vertex); litDir=mul(_World2Ground,float4(litDir,0)).xyz;//把光源方向转换到接收平面空间litDir=normalize(litDir);float4 vt;vt= mul(_Object2World, vertex);vt=mul(_World2Ground,vt);//将物体顶点转换到接收平面空间vt.xz=vt.xz-(vt.y/litDir.y)*litDir.xz;//用三角形相似计算沿光源方向投射后的xzvt.y=0;//使阴影保持在接收平面上//vt=mul(vt,_World2Ground);//back to worldvt=mul(_Ground2World,vt);//返回到世界坐标空间vt=mul(_World2Object,vt);//计算结果重新表达为Object Space坐标return mul(UNITY_MATRIX_MVP, vt);//经典的MVP变换,输出到Clip Space} float4 frag(void) : COLOR {return float4(0.3,0.3,0.3,1);//一个灰色的阴影出来了} ENDCG } }}在此Shader中,我们首先使用固定管线对物体做了一个简单的照明。在计算阴影的ForwardBase中,首先使用一个可以叠加阴影的混合模式,然后使用z偏移保证出来的阴影在接受平面之上。_World2Ground和_Ground2World分别是我们自定义的两个进出阴影接受平面矩阵。在具体计算中,首先将光源方向和投影物体的顶点都转换到接受平面的空间,在它们都处于同一个空间后,通过简单的三角形近似算法,来计算投影物体顶点沿光线投射后在接受平面上的新位置。因为这是一个Build In的Unity Plane,所以只计算其xz分量即可,并将Y分量设为0,这样投影出来的阴影就出现在接受平面上了。投影计算完成后,我们返回到世界空间,然后到投影物体本身的Object Space,之后的事情就如通常那样,一个经典的MVP变换即可。
0 0
- untiy 3d ShaderLab_第9章_1_平面阴影(一)
- untiy 3d ShaderLab_第9章_1_平面阴影(二) 点光源对平面的投影
- untiy 3d ShaderLab_第9章_1_平面阴影(三) 点光源对平面的投影
- untiy 3d ShaderLab_第9章_2_球体阴影(二) 阴影的淡入/淡出
- untiy 3d ShaderLab_第9章_2_球体阴影(一) 平行光对球体的投影
- untiy 3d ShaderLab_第6章_VertexLit渲染路径_1_顶点照明
- untiy 3d ShaderLab_第9章_3_球体阴影(三) 点光源对球体的投影
- untiy 3d ShaderLab_第8章_4_ 单光照贴图在Deferred 渲染路径下的实时阴影
- untiy 3d ShaderLab_第3章_Shader(着色器)中用到的各种空间概念
- untiy 3d ShaderLab_第4章_基本的光照模型
- untiy 3d ShaderLab_第7章_ Forward渲染路径_1_ForwardBase和ForwardAdd
- untiy 3d ShaderLab_第7章_ Forward渲染路径_4_Forward渲染路径总结
- untiy 3d ShaderLab_第8章_ 基于光照贴图的烘焙照明
- untiy 3d ShaderLab_第8章_2_在效果和性能间进行权衡
- untiy 3d ShaderLab_第8章_3_ 单光贴图和Forward 渲染路径
- untiy 3d ShaderLab_第8章_ 基于光照贴图的烘焙照明
- untiy 3d ShaderLab_第 2 章Unity中Shader(着色器)的形态_1_Unity通过ShaderLab 来组织Shader
- untiy 3d ShaderLab_第5章_第一个被执行的Pass_1_不同LightMode 被选择的顺序
- STM8不用手动复位进入自带Bootloader方法(串口下载)
- 第一篇 破壳 实现人物简单的控制的代码和注释 笔记向
- Node.js REPL(交互式解释器)
- java中集合的运用,实现一个简单的购物程序
- Java并发、同步总结
- untiy 3d ShaderLab_第9章_1_平面阴影(一)
- 谈谈期刊论文
- js字符串常用判断方法
- Linux环境配置Lua出现的小问题
- angularjs常见页面查询功能
- 【通知栏】notification
- Node.js 回调函数
- Python3 Scrapy 安装方法 (一脸辛酸泪)
- EK算法网络流模板hdu1532