《Unity着色器和屏幕特效开发秘笈》—— 3.5 使用贴图对模型的高光进行遮罩

简介:

本节书摘来自华章出版社《Unity着色器和屏幕特效开发秘笈》一 书中的第3章,第3.5节,作者:(美)Kenny Lammers,更多章节内容可以访问云栖社区“华章计算机”公众号查看。

3.5 使用贴图对模型的高光进行遮罩

现在,我们已经知道了如何为着色器创建一个高光效果。接下来,我们开始学习如何修改镜面高光,以及更多让最终视觉效果更加艺术的方法。在接下来的章节中,我们将会学到如何使用纹理来改变镜面高光属性和镜面高光强度属性。
现在很多游戏开发渲染管线中都用到了高光贴图技术,因为它可以让3D游戏的美工在每个像素的基础上控制最终的视觉效果。这也为我们提供了一种方式,我们可以在同一个着色器上实现垫型表面和光亮表面,也可以使用另外的贴图来控制镜面高光范围或者镜面高光强度,以实现一个表面是广泛的镜面高光而另一面却是非常尖锐且细小的高光。
很多的效果都可以通过结合两个着色器对纹理进行着色器运算来实现,这也使游戏设计师拥有了控制着色器最终视觉效果的能力,它的关键在于一个高效的渲染管线。接下来看看如何使用纹理来控制镜面反射光照模型。本节将介绍一些新的概念,比如创建自定义Input结构体,还将学习如何从output结构体对数据进行输出,将数据传输至光照函数、Input结构体和surf()函数。了解这些核心的表面着色器元素之间的数据流是一个成功的着色器管线的核心。

3.5.1 准备工作

我们需要一个新的着色器、材质,以及另一个将我们的着色器和材质应用在上面的物体对象。
着色器和材质连接好以后将材质附加在物体对象上,双击着色器脚本,在MonoDevelop 编辑器中打开。
我们还需要使用一个高光贴图。任何贴图都可以,只要它拥有一些不错的色彩和图案变化。下图为本节所使用的贴图:
screenshot

3.5.2 如何操作

1.在Properties块中添加一些新的属性,添加如下代码到着色器Properties块中:
screenshot

2.我们需要在subshader内部中添加相应的变量,这样才能访问Properties块中的属性变量。在#pragma语句的下方添加如下的代码:
screenshot

3.现在我们要添加自定义的Output结构体了。它使我们能够在surf函数和我们的光照函数之间存储更多的数据。现在先不用担心这样做有什么意义。我们将在下一小节对Output结构体的细节进行详细讲解。在SubShader块内部将如下代码放在我们之前声明的变量后面:
screenshot

4.建立好我们的Output结构体之后,需要添加自定义的光照模式。在这里,我们将自定义光照模型命名为LightingCustomPhong。在刚才创建的Output结构体后面输入如下代码:
screenshot

5.为了让自定义的光照模型起作用,我们必须告诉SubShader块需要选择的光照模型。在#pragma语句后面输入下面所示的代码,让着色器加载我们自定义的光照模型:
screenshot

6.因为我们希望通过一张纹理贴图来改变基础高光计算,我们需要专门存储该纹理的另一组UV值。我们通过在Input结构体内存储一个表示UV的变量来实现(UV变量的命名通常是在纹理的变量名前加上“uv_”)。在自定义光照模型后面输入如下代码:
screenshot

7.为了完成我们的着色器,现在我们只需要按照下面的代码修改surf()函数。这一步骤会将纹理信息输入至我们的光照模型中,这样就可以在光照模型函数中使用纹理的每个像素值来修改高光值了。
screenshot

下图展示了使用一个颜色纹理和它的通道信息对我们的高光计算进行遮罩结果。现在对于整个表面我们有了一个变化的高光效果,而不仅仅是一个固定的全局值的高光效果:
screenshot

3.5.3 实现原理

除了使用一个逐像素的纹理改变高光值,并为高光带来更多的视觉感官和深度效果以外,该着色器与Phong模型的计算基本相同。
为了实现这种效果,我们需要做到将表面函数的信息传递给光照函数。因为我们在光照函数内部不能得到一个物体表面的UV。当然你也可以在光照函数内运用程序生成UV,但是如果你想对一个纹理进行解压并得到它的像素信息,就不得不使用Input结构体,而且从Input结构体内访问数据的唯一途径是使用surf()函数。
因此为了建立数据之间的关系,我们需要自定义结构体SurfaceCustomOutput。该结构体作为一个容器可以存储表面着色器中的所有最终数据,而且更重要的是,光照函数和surf()函数都可以访问它的内部数据。所以如果我们自定义了结构体,就可以在结构体里添加更多的数据。下面的代码就是着色器中的SurfaceCustomOutput结构体:
screenshot

然后,我们将它添加到着色器中,而且我们还要告诉surf()函数和光照函数它们使用的Output结构体是我们的自定义结构体,而不是着色器中内置的。通过如下代码来完成这一步骤:
screenshot

现在,我们需要注意surf()函数和光照函数是如何将结构体SurfaceCustomOutput作为它们的参数之一的。我们还在SurfaceOutput结构体中增加了一个叫做SpecularColor的参数。它使我们能够从高光颜色纹理中存储各个像素的信息,并且将它应用到我们的光照函数中,而不仅仅是在高光值上乘以一个单一的全局颜色值。
我们只需使用tex2D()函数就可以得到纹理信息,然后通过将tex2D()函数的返回值赋给o.SpecularColor将纹理信息传递给SurfaceCustomOutput结构体。完成了这些,就可以在光照函数中访问纹理信息了。
screenshot

该技术是你在着色器中创建自定义光照效果的关键。现在,你知道了如何在surf()函数中访问纹理信息并且将它应用到你的光照函数中。这也将帮助你在着色器中创建高质量、逐像素的光照效果。

相关文章
|
5月前
|
算法 vr&ar C#
使用Unity进行虚拟现实开发:深入探索与实践
【8月更文挑战第24天】使用Unity进行虚拟现实开发是一个充满挑战和机遇的过程。通过掌握Unity的VR开发技术,你可以创造出令人惊叹的VR体验,为用户带来前所未有的沉浸感和乐趣。随着技术的不断进步和应用场景的不断拓展,VR开发的未来充满了无限可能。希望本文能为你提供有用的指导和启发!
|
4月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
454 0
|
4月前
|
图形学 iOS开发 Android开发
从Unity开发到移动平台制胜攻略:全面解析iOS与Android应用发布流程,助你轻松掌握跨平台发布技巧,打造爆款手游不是梦——性能优化、广告集成与内购设置全包含
【8月更文挑战第31天】本书详细介绍了如何在Unity中设置项目以适应移动设备,涵盖性能优化、集成广告及内购功能等关键步骤。通过具体示例和代码片段,指导读者完成iOS和Android应用的打包与发布,确保应用顺利上线并获得成功。无论是性能调整还是平台特定的操作,本书均提供了全面的解决方案。
184 0
|
5月前
|
vr&ar 图形学 开发者
步入未来科技前沿:全方位解读Unity在VR/AR开发中的应用技巧,带你轻松打造震撼人心的沉浸式虚拟现实与增强现实体验——附详细示例代码与实战指南
【8月更文挑战第31天】虚拟现实(VR)和增强现实(AR)技术正深刻改变生活,从教育、娱乐到医疗、工业,应用广泛。Unity作为强大的游戏开发引擎,适用于构建高质量的VR/AR应用,支持Oculus Rift、HTC Vive、Microsoft HoloLens、ARKit和ARCore等平台。本文将介绍如何使用Unity创建沉浸式虚拟体验,包括设置项目、添加相机、处理用户输入等,并通过具体示例代码展示实现过程。无论是完全沉浸式的VR体验,还是将数字内容叠加到现实世界的AR应用,Unity均提供了所需的一切工具。
216 0
|
7月前
|
图形学
【unity小技巧】实现FPS武器的瞄准放大效果(UGUI实现反向遮罩,全屏遮挡,局部镂空效果)
【unity小技巧】实现FPS武器的瞄准放大效果(UGUI实现反向遮罩,全屏遮挡,局部镂空效果)
131 1
|
7月前
|
存储 图形学
【unity小技巧】unity中导入下载的3D模型及albedo/baseColor、normal 、AO/Occlus、metallic、roughness贴图纹理设置
【unity小技巧】unity中导入下载的3D模型及albedo/baseColor、normal 、AO/Occlus、metallic、roughness贴图纹理设置
205 0
|
7月前
|
图形学
【用unity实现100个游戏之15】开发一个类保卫萝卜的Unity2D塔防游戏4(附项目源码)
【用unity实现100个游戏之15】开发一个类保卫萝卜的Unity2D塔防游戏4(附项目源码)
114 0
|
5月前
|
图形学 C#
超实用!深度解析Unity引擎,手把手教你从零开始构建精美的2D平面冒险游戏,涵盖资源导入、角色控制与动画、碰撞检测等核心技巧,打造沉浸式游戏体验完全指南
【8月更文挑战第31天】本文是 Unity 2D 游戏开发的全面指南,手把手教你从零开始构建精美的平面冒险游戏。首先,通过 Unity Hub 创建 2D 项目并导入游戏资源。接着,编写 `PlayerController` 脚本来实现角色移动,并添加动画以增强视觉效果。最后,通过 Collider 2D 组件实现碰撞检测等游戏机制。每一步均展示 Unity 在 2D 游戏开发中的强大功能。
286 6
|
5月前
|
图形学 缓存 算法
掌握这五大绝招,让您的Unity游戏瞬间加载完毕,从此告别漫长等待,大幅提升玩家首次体验的满意度与留存率!
【8月更文挑战第31天】游戏的加载时间是影响玩家初次体验的关键因素,特别是在移动设备上。本文介绍了几种常见的Unity游戏加载优化方法,包括资源的预加载与异步加载、使用AssetBundles管理动态资源、纹理和模型优化、合理利用缓存系统以及脚本优化。通过具体示例代码展示了如何实现异步加载场景,并提出了针对不同资源的优化策略。综合运用这些技术可以显著缩短加载时间,提升玩家满意度。
446 5
|
4月前
|
测试技术 C# 图形学
掌握Unity调试与测试的终极指南:从内置调试工具到自动化测试框架,全方位保障游戏品质不踩坑,打造流畅游戏体验的必备技能大揭秘!
【9月更文挑战第1天】在开发游戏时,Unity 引擎让创意变为现实。但软件开发中难免遇到 Bug,若不解决,将严重影响用户体验。调试与测试成为确保游戏质量的最后一道防线。本文介绍如何利用 Unity 的调试工具高效排查问题,并通过 Profiler 分析性能瓶颈。此外,Unity Test Framework 支持自动化测试,提高开发效率。结合单元测试与集成测试,确保游戏逻辑正确无误。对于在线游戏,还需进行压力测试以验证服务器稳定性。总之,调试与测试贯穿游戏开发全流程,确保最终作品既好玩又稳定。
221 4

热门文章

最新文章