近代的游戏大量的采用”全域光照”技术 <br />全域光照(Global illumination,简称GI),是一个用来模拟光的互动和反弹等复杂行为的演算法,要精确的模拟全域光照非常有挑战性,付出的代价也高,正因为如此,现代游戏会先一定程度预先处理这些计算,而非游戏执行时即时运算
同一场景里:没有照明(左),只有直接光源(中),和有间接光源的全域光照(右)的表现,可以注意到颜色如何在不同的表面进行光的”反弹”,产生更真实的结果
在本文中,我们会描述全域光照如何在Unity里运作,带领你透过不同的照明技术解释如何在专案里设定照明,并思考如何透过各种工具帮场景打光
选择一个照明技术
广义的来说,Unity的全域光照是”即时”或是”预先计算好”的,在某些情况下两种方法可以结合使用,照出更逼真的场景
本节我们会针对两种技术的差异优势和使用时机做个简单的描述
即时照明(REALTIME LIGHTING)
预设情况下,Unity的灯源(直接光源,投射灯,点光源)都是即时的,代表这些灯源会把光线照射到场景并以每一帧的频率更新,由于光源是可以在场景内移动的物件,场景灯光的更新是即时的,你可以在游戏视窗和场景视窗看到改变
即时照明的影响:注意到因为没有反射光源的关系阴影是全黑的,只有投射光锥体范围内的物件表面才有光源影响
即时照明是场景里照亮物体最基本的方法,用来照亮角色和会动的物件,可惜的这种照明的光线不会反射,因此我们才导入了全域光照系统,启用了预先计算的技术,都是为了表现一个更逼真的场景
烘焙全域光照(BAKED GI LIGHTING)
当烘焙一张光照贴图(Lightmap)时,场景内的静态物件会基于光的影响算出一张贴图成果,并叠在场景物件之上建立照明效果
左:一个简单的光照贴图场景成果,右:由Unity算出的一张光照贴图(阴影和光源资讯都被纳入计算)
这些”光照贴图”可以包含场景内投射到物体表面的直接光源,以及在不同物体间反射的”间接光源”,这样的光照贴图可以透过物体材质上的著色器(Shader)描述像是颜色的表面资讯(Albedo)和凹凸(Normals)资讯
烘焙光照所产生出来的贴图,是无法在游戏运作的时候变更运算的,因此被定义为静态(Static),虽然仍可在这层贴图上继续叠加光源计算,但两者已无法交互运算
通常我们采用这光照法来让低阶的手机能顺利执行,解决光在游戏中运行的效能问题
预计算全域光照(PRECOMPUTED REALTIME GI LIGHTING)
虽然传统的静态光照贴图无法在游戏执行时改变场景光照条件,但预先计算的即时全域光照系统能帮我们即时运算复杂的场景光源互动
透过这种方法,就能建立昏暗的环境带有丰富的全域光照反射,并即时反映光源的改变,好比做个日晷,阴影的位置和颜色会随着光源移动改变,这在原本的烘焙光照系统是无法达成的
一个用GI呈现的日晷案例
为了在合理的帧率实现这些效果,我们需要在即时运算之前先将一堆垄长的数字资料做”预计算”,预计算负责计算游戏过程中光的复杂行为,它可以在时间空档时进行计算,我们称作一个”离线”运算
如何运作?
最常见的需求是我们希望间接光源能够列入场景光照贴图的计算,幸好,原理上这些间接光源都是从直接光源慢慢转变过来的颜色,只有少部分特定情况有比较大幅度的颜色改变,这样的Unity的全域光照预计算,利用间接光源漫反射(diffuse)特性对运算有利
通常好的阴影是透过即时光源所计算出来的,而非烘焙到光照贴图,假定我们并不需要太复杂的细节取样,可以大大降低全域光照所产生的资料大小
透过预计算来简化整个流程,有效的降低了原本要在游戏中即时计算的全域光照运算数量,如果你要常在游戏中改变光源颜色,旋转光源或是调整光的强度,甚至对场景表面做变更,这点就很重要
Unity从表面上采样底层贴图,并从广义定义颜色的值到一个大型的群组,或是”丛集”,这会产生一个低解析的模拟静态几何,以方便我们用来计算光照
左:场景预览设定为”Albedo”时,可以清楚看到由Unity预计算所产生的丛集,右:如同游戏的场景一样,即时光照计算完结果后套用到场景
基本上,当在计算全域光照时,我们会针对静态场景周围做”光迹追踪”运算,这是非常耗效能的,因此无法苛求要即时运算,相反的,Unity把光迹追踪用在计算这些表面的丛集关系-在预计算”光传输”的阶段,然后把世界串成一个网路结构,我们在关键性的游戏过程就不再需要耗费效能的光迹追踪法
我们有效的创造出一个简化的演算法可以在游戏过程中变化输入结构,这代表我们可以改变光源或是表面颜色,并很快的看到场景内全域光照的影响,算出的结果产出光照贴图透过GPU着色,并和其他照明或是表面混合,最后输出到萤幕上
效益和成本(BENEFITS AND COSTS)
虽然能同时使用烘焙GI和预计算GI,但要注意是同时模拟两个系统,效能负担也会两次运算的总和,不只是因为要储存两套光照贴图在显卡记忆体(VRAM),同时着色器也得付出两次的处理成本
最终要选择哪个方法还是要取决于你专案的性质和预期的硬体考量,例如在手机平台上,效能和记忆体限制较高,烘焙的GI法就会比较适合,如果是在有显卡的电脑上或是游戏机上执行,那可能使用预计算即时全域光照,或两个同时使用就比较可行
决定采用哪一种方法可以针对你的目标平台评估,记得如果专案要同时符合几个不同的硬体需求,往往都是以效能最高的平台为考量
预先计算的过程(THE PRECOMPUTE PROCESS)
在Unity里,预计算是在背景执行,不管是自动流程或是手动启用,计算期间你都可以继续编辑你的游戏物件而不受影响,预计算的时候会在右下角出现一个进度条,不同的演算法会有不同的运算阶段,进度条上方也会显示阶段名称与进度
进度条显示Unity预计算的进度
从上面的例子可以看出,11个工作已经进展到第5个,”丛集”还有108件工作要执行完才会到第6个工作阶段,数值状态列表如下:
启用一个预计算(STARTING A PRECOMPUTE)
只有静态物件会被纳入GI预计算,要让预计算启动首先必须最少要有一个静态物件,不管是单独设定物件或是从层级选单用Shift +选择多个物件后一次修改
从属性面板,将物件的Static勾选起来,这会将该物体所有跟静态物件相关的旗标打开,包含导航旗标或是批次处理旗标,这或许不是你想要的,针对预计算只要把”Lighting Static”这个旗标打勾即可
更细部的控制,只要点选属性介面Static右边的下拉式选单即可,此外,从Window里的Lighting介面也能指定设定静态物件
如果你的场景设为自动(Lighting->Scene->Auto),Unity的预计算就会自动启动,否则就需要用下列的流程手动执行
自动/手动预计算(AUTO/MANUAL PRECOMPUTE)
假如Lighting介面底下Auto这个选项是被勾选的(Lighting->Scene->Auto),那么预计算就会自动在背景不停的改变场景产出的静态几何
但如果这个勾选没勾,你将需要点击在Auto旁边的”Build”按钮手动启动预计算,这会用同样的方式进行预计算,让你比较好控制计算的开始时间
手动启动预计算会对场景所有的照明与各方面进行重新评估并重新计算,如果你希望有选择性的计算,可以从”Build”旁边的下拉选单来选择
使用烘焙GI或是预计算GI(ENABLING BAKED GI OR PRECOMPUTED REALTIME GI)
预设情况下,两种计算法在Unity里都是启用状态(Lighting->Scene),因为如此,你可以针对单独的光源设定要采用哪种计算法(Inspector->Light->Baking)
在一个场景同时采用两种方法可能会对效能造成负担,最好的做法在同一个时间只用一种方法,要关闭任何一种方法可以从GI的介面(Lighting->Scene),把不要用的方法取消打勾即可,只有有打勾的算法会计算,任何相关光的设定都会被覆盖
预灯光设定(PRE-LIGHT SETTINGS)
Unity里每盏灯光预设的烘焙模式都是”Realtime”,这代表这些灯光仍然会照亮你的场景,Unity的预计算GI系统会处理间接光
但如果预设的烘焙模式是”Baked”,那么这些灯光将会透过Unity的烘焙GI系统处理直接光源和间接光源,产生出来的光照贴图一旦贴到场景上在执行期间是不能改变的
一个设定烘焙模式为”Realtime”的点光源
选择烘焙模式为”Mixed”的话,场景内的静态物件会被烘焙GI拿去做计算,然而,不像”Baked”模式,混合模式的灯光仍会继续运算即时光源到非静态物件上,这对于你想把静态环境烘成光照贴图,但同时又希望同样一盏灯能为会动的角色计算阴影很有帮助
GI快取(GI CACHE)
无论是烘焙还是预计算系统,Unity会”缓存”场景的光照资料到”GI快取”,并会在计算时尝试重复运用这些数据来节省时间,你对场景的改变会影响这个数据重复利用的多寡
如果你要清除这个快取可以从(Preference->GI Cache->Clear Cache)来清除,清除后代表所有资料都必须重新运算,因次会花费一些时间,在某些情况下你也许需要降低档案空间大小(例如要把专案转到另外一台电脑)是有帮助的
场景设定(SCENE SETUP)
选择着色路径(CHOOSING A RENDERING PATH)
Unity支援许多渲染技术或”路径”,在启动一个专案时,必须要订出出一个路线,Unity预设是”Forward Rendering”
在”Forward Rendering”里,每个物件渲染是根据每个影响物件的光,透过”Pass”来渲染,所以有可能一个物件被重复渲染了好几次,取决于有几盏灯在作用范围里
这种方法的优点是快速,也代表硬体需求低,此外,这种正向渲染提供了广泛的自订”着色模型”,可以快速处理透明度,也支援像是多重采样柔边(MSAA)的硬体功能,等等有些在其他路径上是无法实现的功能,对于图形品质有很大的影响
然而他的缺点是要为每盏灯光付出相对应的成本,也就是说,物件被越多盏灯光影响,花费的运算成本就越高,有些类型的游戏必需要大量的光源,就会令人望之却步,反观如果你能管理好你的灯光数量,那这个路径会是一个非常快速的解决方案
“Deferred”路径,是延迟了光的遮蔽与混合资讯直到第一次接收到的表面的位置法线以及材质资料渲染到一个”几何缓冲器”(G-buffer)作为一个萤幕空间的贴图,最后合成这些结果,这种方法优点是照明的渲染成本是和像素数量成正比,而非灯光数量,因此你不用再管控场景灯光数量,某些游戏类型将会是一个关键优势
“Deferred”路径呈现可预见的效能特点,但通常需要较强大的硬体,对于手机平台支援度也较低
关于”Deferred”,”Forward”或是其他路径的更多资讯,可以参阅这里 (英文连结)
选择一个色彩空间(CHOOSING A COLOR SPACE)
除了 要选好渲染路径之外,一开始选择一个”色彩空间(Color Space)”也是很重要的事情,色彩空间决定采用哪种演算法来计算照明或材质载入时的颜色混合,这会对游戏的画面真实感有很大的影响,但大多数情况下,太超过的色彩空间设定可能会被目标平台的硬体强制限制
推荐比较接近真实的色彩空间是Linear, 你可以从Player Setting里面找到”Color Space”来设定(Edit->Project Setting->Player)
设定Linear的优点是会让场景内的提供给着色器的颜色也会因为光强度增加变亮,如果换成”Gamma”色彩空间,亮度马上会转为以白色做为参考,这将不利于图像的品质
采用Linear和Gamma颜色空间的图像对照表,可以注意到切换成Gamma时颜色快速的换成以白色为光照强度为基准
Linear另一个好处是著色器能在没有Gamma补偿的情况下对贴图进行取样,这有助于确保颜色品质在经过渲染管道还能保持一致性,能提高色彩和计算的精度,最后萤幕的输出结果更为真实
可惜的是Linear颜色空间有些手机平台甚至有些游戏机不支援,应该说PC或是一些新手机硬体和次世代游戏机才会支援Linear颜色空间,在这种情况下,就得用Gamma方法替代了
确认你的目标平台是哪一种之后才选择适合的颜色空间是很重要的,想要了解更多关于颜色空间的资讯,可以参阅这里 (英文)
高动态范围(HDR)
如同颜色空间,相机的”Dynamic range”动态范围是需要指定的,从本质上来说,这是用来定义相机截取的最亮与最暗的颜色范围,要启用HDR可以从相机元件里把HDR项目打勾即可,要注意的是,某些手机硬体是不支援HDR的,在渲染路径为Forward Rendering并启用MSAA柔边时也不支援HDR
HDR和Linear颜色空间一起搭配是最好的,在处理非常明亮的颜色的时候还能保有准确性
在预设的情况下,Unity是使用低动态范围(LDR),颜色会以每个频道8位元储存三原色(红绿蓝),8位元确切的意思是用8个0或1的数字组合成的值,一共会有256种组合,三个频道256 x 256 x 256可以组合出1600万种色采来表达从黑色到白色不同层度的灰阶
在现实环境中,颜色远远超出1600万个色阶,颜色和亮度的排列甚至远远超越了我们人眼所能辨识的,同样的,Unity能够处理超出这些范围的颜色并输出高于LDR设备支援的颜色,用来提供给像是电脑画面的高品质结果,尽管现今输出装置仍有很多限制,这些多出来的数值仍然可有很多的应用
启用HDR之后,会储存更大精度的颜色资料(使用浮点运算表示),能处理更多更亮的颜色范围
HDR能让我们在同一画面下维持两个巨大差异的亮度,例如场景的阴影区域和户外的亮光,我们也可以在环境场景建立一个”光晕”或发光特效,透过这些特效或可视的光影效果能增加画面的真实感,但也要小心处理这些效果以防止他们曝光过度
色调映射(TONEMAPPING)
以摄影来比喻的话,如果我们使用不同的曝光度来拍摄我们的场景,我们可以先观察到哪些颜色会因为曝光过度而遗失,浅色调在很亮的区域可能被白色给覆盖过去,暗色调可能会被黑色给覆盖,这像是电脑图学的”色调映射”,当颜色在显示设备(比如电脑萤幕)表现范围之外时,演算法会将颜色修正为装置合理的颜色并重现在萤幕上
当相机启用HDR时,建议相机加入Tonemapping这只程式(Assets->Import Package->Effects),这只程式可以帮你转换控制超出范围的颜色成为合理的颜色
更多关于色调映射的资料可以查看这里 (英文)
环境光(Ambient Lighting)
场景中一个照亮整体环境非常重要的就是”环境光”,可以说是影响场景光源最全面的一个要素
环境光很多情况都很有用,也取决于你所选的风格,比如卡通风格的阴影不清楚或灯光是手绘风格,环境光也很适合用在当你不想单独调整场景内的灯光但又想要增加整体场景亮度的时候
在没有使用Unity 5全域光照的功能时,环境光不会算出准确的物理遮挡,但如果开启了任何一种GI的情况下,从Skybox照下来的环境光就能被算出遮挡,结果会更加真实
在同一个场景之下,左边是没有任何光源的场景,右边则开启了环境光,可以注意到天空盒的亮度并没有因为调整环境光的强度而改变
预灯光设定(PRE-LIGHT SETTINGS)
Unity里每盏灯光预设的烘焙模式都是”Realtime”,这代表这些灯光仍然会照亮你的场景,Unity的预计算GI系统会处理间接光
但如果预设的烘焙模式是”Baked”,那么这些灯光将会透过Unity的烘焙GI系统处理直接光源和间接光源,产生出来的光照贴图一旦贴到场景上在执行期间是不能改变的
一个设定烘焙模式为”Realtime”的点光源
选择烘焙模式为”Mixed”的话,场景内的静态物件会被烘焙GI拿去做计算,然而,不像”Baked”模式,混合模式的灯光仍会继续运算即时光源到非静态物件上,这对于你想把静态环境烘成光照贴图,但同时又希望同样一盏灯能为会动的角色计算阴影很有帮助
GI快取(GI CACHE)
无论是烘焙还是预计算系统,Unity会”缓存”场景的光照资料到”GI快取”,并会在计算时尝试重复运用这些数据来节省时间,你对场景的改变会影响这个数据重复利用的多寡
如果你要清除这个快取可以从(Preference->GI Cache->Clear Cache)来清除,清除后代表所有资料都必须重新运算,因次会花费一些时间,在某些情况下你也许需要降低档案空间大小(例如要把专案转到另外一台电脑)是有帮助的
场景设定(SCENE SETUP)
选择着色路径(CHOOSING A RENDERING PATH)
Unity支援许多渲染技术或”路径”,在启动一个专案时,必须要订出出一个路线,Unity预设是”Forward Rendering”
在”Forward Rendering”里,每个物件渲染是根据每个影响物件的光,透过”Pass”来渲染,所以有可能一个物件被重复渲染了好几次,取决于有几盏灯在作用范围里
这种方法的优点是快速,也代表硬体需求低,此外,这种正向渲染提供了广泛的自订”着色模型”,可以快速处理透明度,也支援像是多重采样柔边(MSAA)的硬体功能,等等有些在其他路径上是无法实现的功能,对于图形品质有很大的影响
然而他的缺点是要为每盏灯光付出相对应的成本,也就是说,物件被越多盏灯光影响,花费的运算成本就越高,有些类型的游戏必需要大量的光源,就会令人望之却步,反观如果你能管理好你的灯光数量,那这个路径会是一个非常快速的解决方案
“Deferred”路径,是延迟了光的遮蔽与混合资讯直到第一次接收到的表面的位置法线以及材质资料渲染到一个”几何缓冲器”(G-buffer)作为一个萤幕空间的贴图,最后合成这些结果,这种方法优点是照明的渲染成本是和像素数量成正比,而非灯光数量,因此你不用再管控场景灯光数量,某些游戏类型将会是一个关键优势
“Deferred”路径呈现可预见的效能特点,但通常需要较强大的硬体,对于手机平台支援度也较低
关于”Deferred”,”Forward”或是其他路径的更多资讯,可以参阅这里 (英文连结)
选择一个色彩空间(CHOOSING A COLOR SPACE)
除了 要选好渲染路径之外,一开始选择一个”色彩空间(Color Space)”也是很重要的事情,色彩空间决定采用哪种演算法来计算照明或材质载入时的颜色混合,这会对游戏的画面真实感有很大的影响,但大多数情况下,太超过的色彩空间设定可能会被目标平台的硬体强制限制
推荐比较接近真实的色彩空间是Linear, 你可以从Player Setting里面找到”Color Space”来设定(Edit->Project Setting->Player)
设定Linear的优点是会让场景内的提供给着色器的颜色也会因为光强度增加变亮,如果换成”Gamma”色彩空间,亮度马上会转为以白色做为参考,这将不利于图像的品质
采用Linear和Gamma颜色空间的图像对照表,可以注意到切换成Gamma时颜色快速的换成以白色为光照强度为基准
Linear另一个好处是著色器能在没有Gamma补偿的情况下对贴图进行取样,这有助于确保颜色品质在经过渲染管道还能保持一致性,能提高色彩和计算的精度,最后萤幕的输出结果更为真实
可惜的是Linear颜色空间有些手机平台甚至有些游戏机不支援,应该说PC或是一些新手机硬体和次世代游戏机才会支援Linear颜色空间,在这种情况下,就得用Gamma方法替代了
确认你的目标平台是哪一种之后才选择适合的颜色空间是很重要的,想要了解更多关于颜色空间的资讯,可以参阅这里 (英文)
高动态范围(HDR)
如同颜色空间,相机的”Dynamic range”动态范围是需要指定的,从本质上来说,这是用来定义相机截取的最亮与最暗的颜色范围,要启用HDR可以从相机元件里把HDR项目打勾即可,要注意的是,某些手机硬体是不支援HDR的,在渲染路径为Forward Rendering并启用MSAA柔边时也不支援HDR
HDR和Linear颜色空间一起搭配是最好的,在处理非常明亮的颜色的时候还能保有准确性
在预设的情况下,Unity是使用低动态范围(LDR),颜色会以每个频道8位元储存三原色(红绿蓝),8位元确切的意思是用8个0或1的数字组合成的值,一共会有256种组合,三个频道256 x 256 x 256可以组合出1600万种色采来表达从黑色到白色不同层度的灰阶
在现实环境中,颜色远远超出1600万个色阶,颜色和亮度的排列甚至远远超越了我们人眼所能辨识的,同样的,Unity能够处理超出这些范围的颜色并输出高于LDR设备支援的颜色,用来提供给像是电脑画面的高品质结果,尽管现今输出装置仍有很多限制,这些多出来的数值仍然可有很多的应用
启用HDR之后,会储存更大精度的颜色资料(使用浮点运算表示),能处理更多更亮的颜色范围
HDR能让我们在同一画面下维持两个巨大差异的亮度,例如场景的阴影区域和户外的亮光,我们也可以在环境场景建立一个”光晕”或发光特效,透过这些特效或可视的光影效果能增加画面的真实感,但也要小心处理这些效果以防止他们曝光过度
色调映射(TONEMAPPING)
以摄影来比喻的话,如果我们使用不同的曝光度来拍摄我们的场景,我们可以先观察到哪些颜色会因为曝光过度而遗失,浅色调在很亮的区域可能被白色给覆盖过去,暗色调可能会被黑色给覆盖,这像是电脑图学的”色调映射”,当颜色在显示设备(比如电脑萤幕)表现范围之外时,演算法会将颜色修正为装置合理的颜色并重现在萤幕上
当相机启用HDR时,建议相机加入Tonemapping这只程式(Assets->Import Package->Effects),这只程式可以帮你转换控制超出范围的颜色成为合理的颜色
更多关于色调映射的资料可以查看这里 (英文)
环境光(Ambient Lighting)
场景中一个照亮整体环境非常重要的就是”环境光”,可以说是影响场景光源最全面的一个要素
环境光很多情况都很有用,也取决于你所选的风格,比如卡通风格的阴影不清楚或灯光是手绘风格,环境光也很适合用在当你不想单独调整场景内的灯光但又想要增加整体场景亮度的时候
在没有使用Unity 5全域光照的功能时,环境光不会算出准确的物理遮挡,但如果开启了任何一种GI的情况下,从Skybox照下来的环境光就能被算出遮挡,结果会更加真实
在同一个场景之下,左边是没有任何光源的场景,右边则开启了环境光,可以注意到天空盒的亮度并没有因为调整环境光的强度而改变
透过将物件设定为静态物件来启用全域光照的场景,可以注意到光线在不同的表面有遮挡的效果
使用环境光的好处是耗用效能很低,因此在手机平台上只要灯光数量控制得宜也能得到很好的表现
你可以从Lighting视窗(Window->Lighting)找到Environment Lighting设定区域(Lighting->Scene),改变Ambient Source来改变环境光来源
Ambient Source的预设值是”Skybox”,上图的天空盒是Unity 5系统产生的一个预设天空盒,带有一个蓝色调的环境光,以及一些用该半球体上纯色与渐变色的设定
要注意的是改变Lighting里面的Ambient Source并不会让Skybox的颜色改变,但会影响到场景内环境光照射
反射源(REFLECTION SOURCE)
预设的情况下,场景中的物件会使用Unity内建的标准着色器(Standard Shader)来渲染,标准着色器是一个基于物理的著色器(PBS),它会透过模仿真实世界的物理特性像是反射或能量传递等来模拟真实世界里光在材质上的表现
当使用标准着色器时,每一个材质都会具有一定程度的镜面反射(specularity)和金属反射(metalness)属性,在没有强大的硬体来处理即时光迹追踪反射的情况下,我们得仰赖预先计算渲染反射,我们使用了一个由六张描述天空的图片所组成的方体贴图(Cubemap)或从Unity 5用来从定点搜集环境资讯的反射探头(Reflection Probe)产生所需的贴图,然后在和其他光和地表资讯混合运算来模拟如同我们真实世界看到的反射效果
在预设的情况下,调高Specular和Metalness会更清楚的反射天空盒,反射的来源可以从设定调整
场景内的物件在预设的情况下会反射天空盒的内容,但你可以从Lighting介面里找到Reflection Source属性来改变来源,指定一个新的方体贴图,或指定一个反射探头来定义
反射探头(REFLECTION PROBES)
天空盒的资讯不可能包含所有的场景物件,在许多情况下,物件从天空搜集反射资讯时可能会被遮蔽,像是室内物件或是在类似桥或是隧道等建筑物里的物件,为了要准确反射这些物件,我们必须用反射探头针对这些物件取样,这种探头从他们的位置对周围取样并把结果写到方体贴图,可以让周围经过的物体得到环境的反射影像
你可以透过GameObject->Light->Reflection Probe来新增一个反射探头
反射探头的位置会决定方体贴图取样的内容,以及反射所看起来的样子,一般来说,基于效能考量反射探头越少越好,请记住,反射探头并非用来让物理得到精确结果,而是让游戏世界有更好的反射,大多数情况下几个安排妥当的反射探头就很足够了
左图:场景只有预设的反射设定右图:场景加入了反射探头后的结果
在反射探头的属性面板我们可以设定Type为即时(Realtime),烘焙(Baked)或自订(Custom),需要明白的是,即时反射的设定对效能极为不利,每多一个即时反射探头就会多出额外六次的渲染运算,因此摆设即时的反射探头应该要有明确的需求,例如反射会闪动的霓虹灯,否则一般来说建议设成烘焙的就够了,效能也会好很多
需要注意的是,只有标记为”Reflection Probe Static”的物件才会被反射探头取样,从属性介面上静态物件(Static)的下拉选单打勾即可,相反的,即时的反射探头会对所有可见的物体取样,除非你在遮罩(Mask)选单指定剔除它
照亮一个场景(LIGHTING A SCENE)
我们已经介绍了一些在Unity里针对场景照明开始工作之前所需要考虑的条件,希望你对目标平台该用哪些设定已经有了一个方向(手机平台采用烘焙GI和Gamma颜色空间,PC或游戏机采用即时GI和Linear颜色空间)
接下来让我们来看看有哪些协助制定光源的工具
定向光源(DIRECTIONAL LIGHTS)
“定向光”非常适合用来模拟阳光,它的特性就像是个太阳,定向光能从无限远的距离投射光源到场景
从定向光发出来的光线是互相平行的,也不会像其他种光源会分岔,结果就是不管物件离定向光源多远,投射出来的阴影看起来都一样,这其实对户外场景的照明很有利
定向光没有真正的光源座标,放置在场景任何地点都不会影响光的效果,只有旋转会影响定向光的照射结果
其他有光源座标的灯光类型,例如投射灯(Spotlights),角色阴影会因为接近或远离光源而改变,这也许在照亮室内环境时会是个问题,一般来说,避免角色太接近隐形的光源,我们会建立一个亮点来假装光源
使用定向光不用考虑距离,不管多远它都会影响场景所有的表面(除非被剔除),当使用延迟(Deferred)渲染路径时会造成一些效能损耗,要注意的是,使用这个渲染路径时,光的效能代价和他影响的像素数目是成正比的,但虽然需要消耗效能,起码结果较为统一,因此比较容易调整平衡
在预设情况下,新的场景都会附带一盏定向光,在Unity 5里还会与天空盒系统关联(Lighting->Scene->Skybox),你也可以删除预设的定向光并创建一个新的光源,然后从Sun这个属性重新指定(Lighting->Scene->Sun)
旋转预设的定向光会导致天空盒也跟着更新,如果光的角度和地面平行就可以做出日落的效果,把光源转到天空导致变黑就能做出夜晚的效果,从上往下照就会模拟日间的效果
如果天空盒有指定为环境光源(Ambient Source),那么天空盒的颜色就会影响环境里面的物件
点光源(POINT LIGHTS)
点光源可以想像是在3D空间里一个对着所有方向发射光线的点,很适合用来制作像是灯泡,武器发光或是从物体发射出来的爆炸效果
点光源的亮度从中心最强一直到范围属性(Range)设定的距离递减到0为止,光的强度从光源到距离成反比,这是所谓的”平方反比定律”,类似光在现实世界的行为
点光源从它的位置对四面八方射出光线,球形的小图示代表光的”范围”,光线到达此范围是会”衰减”到0,但如果有间接光源或反射光则会继续投射
点光源开启阴影运算是很耗效能的,因此必须谨慎使用,点光源的阴影为了要给六个不同的世界方向会运算六次,在比较差的硬体开启此功能会造成较大的效能负担
当在场景中加入点光源时要注意,目前它们不支援阴影的间接反射,这代表由点光源产生的光线,只要在距离内有可能会穿过物件反射到另外一面,这可能会导致墙壁或地板”漏光”,因此放置点光源要格外注意,然而如果是采用Backed GI的话,就不会有这类的问题产生
聚光灯(SPOTLIGHTS)
聚光灯投射一个锥体在他的Z轴前方,这个锥体的宽度由投射角度(Spot Angle)属性控制着,光线会从源头到设定的范围慢慢衰减到0,同时越靠近锥体边缘也会衰减,把投射角度的值加大会让锥体宽度加大,同时也让边缘淡化的力度变大,这现象学名叫做”半影”
聚光灯有许多用途,他们可以用来模拟路灯,壁灯,或许多创意用法,例如模拟手电筒,因为投射区域能精确的控制,因此很适合用来模拟打在角色身上的光或是模拟舞台灯光效果等等
光线会因为离源头越远而递减,可以注意到光也会因为越靠近锥体边缘而变弱,我们称之为半影区,这会因为锥体角度变大而更明显
和点光源一样,使用预计算GI时,聚光灯不支援间接光阴影,这表示灯光会穿过几何影响到另外一面,因此放置投射灯要特别注意
区域光(AREA LIGHTS)
区域光可以当作是摄影用的柔光灯,在Unity里面他们被定义为单面往Z轴发射光线的矩形,目前只能和烘焙GI一起使用,区域光会均匀的照亮作用区域,虽然区域光没有范围属性可以调整,但是光的强度也是会随着距离光源越远而递减
区域光照亮表面并在区间产生漫反射与柔和的阴影
区域光用在建立柔和的照明效果非常有用,光线在任何方向穿过光的表面时会产生不同方向的折射-造成在物件上产生漫反射,常见的用途是拿来当作天花板壁灯或是背光灯
为了实现这功能,我们必须从每个光照贴图像素上发射一定数量的光线,背对着区域光以确定光有能见度,这代表区域光的计算是消耗很大的,而且会延长烘焙的时间,但如果运用得宜可以增加场景光的深度,那么消耗就很值得,值得注意的是区域光只能用在烘焙,因此不影响游戏效能
发光材质(EMISSIVE MATERIALS)
虽然区域光不支援即时GI,Unity提供另外一个柔和的灯光效果叫做发光材质(Emissive Materials),和区域光一样,发光材质可以让物体表面发光,他们可以反射场景内像是颜色或是光强度等等能在游戏内改变的光源
自发光(Emission)是一个在标准着色器(Standard Shader)内的属性,允许静态物件成为一个发光体,预设情况下是0,代表指定了这个材质并不会有任何的自发光反应,HDR颜色选择器能指定发光颜色,强度能在0-1的范围调整,来建立类似区域光的效果
发光材质并没有范围属性,但从材质发出的光会以二的次方速度递减,自发光材质只会作用在有标记为”Static”或”Lightmap Static”标签的物件,同样的,如果发光材质附加在非静态物件或像是角色的动态物件则不会有任何作用
然而,材质设定只要emission数值大于0,即使他们不接收场景光源在画面上也会有发光的效果,这种效果也可以透过将emission属性底下的”Global Illumination”改为”None”,像这样的自发光材质很适合来模拟霓虹灯等类似的光源
使用Unity标准着色器并附加自发光材质的一个范例,注意从Unity Logo的自发光也会计算阴影,在这个案例让球体有了阴影
发光材质只会影响场景内的静态物件,如果你想要影响像是角色的动态物件,就必须采用光探头系统(Light Probes),在游戏周期改变发光材质的值会更新光探头取样,并直接在结果上看到变化
光探头(LIGHT PROBES)
静态物件只被Unity全域光照GI系统计算,为了使动态物件能够和静态场景接收到的光影资讯互动,我们需要纪录这些光的资讯并做成可以在执行期间快速存取的格式
我们在场景放置许多取样点来截取各个 方向搜集来的资讯,颜色资讯会被编成能在游戏中快速被取出的一组数值(或系数),这些取样点我们称为”光探头”
使用了光探头的场景,注意图中光探头放置位置在光线容易变化的地方,例如阴影或是颜色转换的地方
光探头允许移动物件接受由全域光照GI所计算出来复杂的反射光源,物件在渲染网格的时候会判断附近光探头的位置并且把光的资讯一并融合计算,这是透过找寻由光探头所产生的一个四面体,然后决定哪个四面体的落入物件的轴向,这样就能让场景内的动态物件正确地接受光资讯,如果没有放置光探头,动态物件就无法接受全域光照的资讯,造成动态物件比场景还要暗
预设的情况下,场景是没有任何光探头的,你必须从GameObjects->Light->Light Probe Group自行建立光探头群组
假如全域光照里的Auto是打勾的(Lighting->Scene->Auto),当光源或是静态物件更新时,光探头资讯也会即时更新,没打勾的话必须点Build运算才会更新
接下来...
本篇文章大致描述了Unity的光源运作基础原理,但是还有许多细节没有涵盖到,后面还有两篇文章会陆续推出,请密切注意