【URP】[投影Projector]解析与应用
Unity中的投影系统主要包括传统Projector组件和URP Decal Projector两种实现方式。
【从UnityURP开始探索游戏渲染】专栏-直达
发展历史与技术演进
- 传统Projector组件:Unity早期内置的通用投影系统,通过摄像机空间矩阵计算实现材质投影,但存在性能开销大、与URP管线兼容性差的问题
- URP Decal Projector:自URP 12(Unity 2021)引入的专用贴花系统,采用延迟渲染路径和屏幕空间计算,性能优化显著
核心实现原理
传统Projector:
- 继承自Behaviour的UnityEngine.Projector类实现
- 基于视锥体裁剪和投影矩阵变换
- 通过MaterialPropertyBlock动态修改着色器参数
URP Decal Projector:
- 使用DecalRendererFeature扩展URP管线
- 采用Deferred Decals技术,在GBuffer阶段后处理
- 通过DecalProjector组件管理投影参数
- Projector组件的工作只是检测了所有范围内的模型,并传递了投影空间矩阵。
- 投影着色器,用这个投影矩阵,左乘齐次裁剪空间的点,得到的坐标作为uv坐标传递给片元着色器。
- 片元着色器中使用tex2Dproj(_MainTex, i.texc)获取实际纹理。这里的tex2Dproj实际上是将 $i.texc = i.texc/i.texc.w$ 透视除法
- tex2D(_MainTex, i.texc) 获取纹理文素
实际应用对比
传统Projector示例:
csharp
// 动态调整投影参数
Projector proj = GetComponent<Projector>();
proj.nearClipPlane = 0.5f;
proj.material.SetFloat("_Falloff", 0.8f);
URP Decal Projector工作流:
- 创建Decal Material:使用Shader Graphs/Decal着色器
- 配置Decal Projector:调整Size/Depth等参数
- 通过脚本动态生成:
csharp
// 在碰撞点生成弹孔
DecalProjector CreateBulletHole(Vector3 position) {
GameObject decal = Instantiate(decalPrefab, position, Quaternion.LookRotation(-hit.normal));
return decal.GetComponent<DecalProjector>();
实践
典型应用场景:
URP Decal Projector基础配置
- 环境准备:
- 创建URP Asset并配置到Graphics Settings
- 通过Package Manager安装Universal RP 12.1.6+版本
- 核心组件添加:
csharp
// 添加Decal Renderer Feature
var renderer = URP资产中的Renderer数据;
renderer.AddFeature(new DecalRendererFeature());
环境细节:墙面涂鸦、地面污渍(URP Decal)
- 墙面涂鸦实现:
- 创建Decal Material:使用Shader Graphs/Decal着色器,配置Albedo贴图为涂鸦图案
- 调整Projector参数:Size=(2,2,0.5), Fade Factor=0.8
- 层过滤:设置Affected Layers仅包含Wall层
- 地面污渍效果:
材质配置:混合Albedo(污渍贴图)和Normal Map(凹凸细节)
投影参数:Depth=0.3, Angle Fade=(0.8,1.0)
动态生成代码:
csharp
void CreateMudDecal(Vector3 position) {
var decal = Instantiate(decalPrefab);
decal.transform.position = position + Vector3.up*0.1f;
decal.GetComponent<DecalProjector>().size =
new Vector3(Random.Range(0.5f,1.5f), 0.2f, Random.Range(0.5f,1.5f));
}
动态效果:弹孔、血迹(两种方案均可)
弹孔效果高级实现:
预制体配置:包含Decal Projector和Particle System
命中点生成逻辑:
csharp
void CreateBulletHole(RaycastHit hit) {
var decal = Instantiate(bulletHolePrefab,
hit.point + hit.normal*0.01f,
Quaternion.LookRotation(-hit.normal));
decal.transform.Rotate(Vector3.forward, Random.Range(0,360));
Destroy(decal, 10f);// 10秒后自动消失
}
血迹效果优化方案:
材质混合:使用Multiply混合模式增强真实感
动态渐隐控制:
csharp
IEnumerator FadeDecal(DecalProjector decal, float duration) {
float startTime = Time.time;
while(Time.time < startTime + duration) {
float t = (Time.time - startTime) / duration;
decal.fadeFactor = Mathf.Lerp(1, 0, t);
yield return null;
}
Destroy(decal.gameObject);
}
特殊光照:投影阴影(传统Projector)
性能优化:
- 传统Projector:限制影响范围,使用简单材质
- URP Decal:
- 启用Layer Mask过滤,控制Fade Distance
- 批处理设置:对静态Decal启用Batching
- 视锥体裁剪:调整Projector的Far Clip Plane避免过度绘制
- 动态控制:非可见区域暂停Decal更新
- 材质复用:相同类型效果共享材质实例
- 实际项目建议采用分层管理系统,对不同类型Decal进行分类管理,例如:
- 永久性Decal(涂鸦):使用单独Layer并启用Occlusion Culling
- 临时性Decal(弹孔):使用对象池技术减少实例化开销
- 环境Decal(污渍):烘焙到Lightmap中减少运行时消耗
参数配置要点:
- 投影角度:通过旋转控制投影方向
- 衰减控制:使用Falloff参数平滑边缘
- 混合模式:Alpha Blend/Multiply选择
当前URP项目推荐优先使用Decal Projector系统,其性能表现和视觉效果更优,特别是在需要大量动态贴花的场景中。传统Projector仍适用于需要实时阴影投影等特殊需求场景.
【从UnityURP开始探索游戏渲染】专栏-直达
(欢迎点赞留言探讨,更多人加入进来能更加完善这个探索的过程,)
【URP】[投影Projector]解析与应用的更多相关文章
- threejs三维地图大屏项目分享
这是最近公司的一个项目.客户的需求是基于总公司和子公司的数据,开发一个数据展示大屏. 大屏两边都是一些图表展示数据,中间部分是一个三维中国地图,点击中国地图的某个省份,可以下钻到省份地图的展示. 地图 ...
- Gis数据处理
几何投影和解析投影几何投影是将椭球面上的经纬线网投影到几何平面上,然后将几何面展为平面.几何投影可以分为方位投影.圆柱投影和圆锥投影.这三种投影纬线的形状不同.方位投影纬线的形状是同心圆:圆柱投影纬线 ...
- Unity3D手游开发日记(1) - 移动平台实时阴影方案
阴影这个东西,说来就话长了,很多年前人们就开始研究出各种阴影技术,但都存在各种瑕疵和问题,直到近几年出现了PSSM,也就是CE3的CSM,阴影技术才算有个比较完美的解决方案.Unity自带的实时阴影, ...
- Trident中的解析包含的函数操作与投影操作
一:函数操作 1.介绍 Tuple本身是不可变的 Function只是在原有的基础上追加新的tuple 2.说明 如果原来的字段是log,flag 新增之后的tuple可以访问这些字段,log,fla ...
- 使用QT开发GoogleMap瓦片显示和下载工具(2)——Google地图瓦片投影和获取网址解析
这篇主要说的是Google地图采用的投影方式,瓦片划分方式以及如何从给定的经纬度获取瓦片的数据的网址.所有资料均来自于网络,并亲自试验可行. Google地图投影 首先是地图投影问题,因为地球本身是一 ...
- PL-SVO公式推导及代码解析:地图点重投影和特征对齐
对当前帧进行地图点重投影和特征对齐 // map reprojection & feature alignment SVO_START_TIMER("reproject") ...
- SVO原理解析
最近空闲时间在研究Semi-Direct Monocular Visual Odometry(SVO)[1,2],觉得它值得写一写.另外,SVO的运算量相对较小,我想在手机上尝试实现它. 关于SVO的 ...
- [CG编程] 基本光照模型的实现与拓展以及常见光照模型解析
0.前言 这篇文章写于去年的暑假.大二的假期时间多,小组便开发一个手机游戏的项目,开发过程中忙里偷闲地了解了Unity的shader编写,而CG又与shaderLab相似,所以又阅读了<CG教程 ...
- Google Maps地图投影全解析(3):WKT形式表示
update20090601:EPSG对该投影的编号设定为EPSG:3857,对应的WKT也发生了变化,下文不再修改,相对来说格式都是那样,可以到http://www.epsg-registry.or ...
- Spark之SQL解析(源码阅读十)
如何能更好的运用与监控sparkSQL?或许我们改更深层次的了解它深层次的原理是什么.之前总结的已经写了传统数据库与Spark的sql解析之间的差别.那么我们下来直切主题~ 如今的Spark已经支持多 ...
随机推荐
- DRF案例
1 反序列化更新,instance 就传要修改的对象,保证修改完成 def update(self, instance, validated_data): publish_id = validated ...
- 在华为云服务器上测试GCC for OpenEuler的特性
前言 操作系统课程任务 探讨 GCC for openEuler 的特性和优势 什么是 GCC for openEuler? GCC for openEuler 基于开源 GCC-10.3 版本(GC ...
- 多Agent协作入门:群组聊天-AgentGroupChat
大家好,我是Edison. 近日抽空学习了下Semantic Kernel提供的AgentGroupChat对象写了一个多Agent群组对话的Demo,总结一下分享与你.当然,多Agent协作还有其他 ...
- 极大提高项目部署的生产力!分享一个半自动化的CICD实现方案
前言 完全自动化的 CICD 确实好,代码提交后就自动构建自动发布新版本,实现不停机更新的情况下,还能随时回滚,这搁谁不喜欢啊~ 但理想很丰满,现实往往很骨感,不是所有开发/生产环境都具备部署 CIC ...
- doget高速下载 github release
平常下载github release或者国外文件时,可能遇到下载速度慢,不能下载等情况. 第一步 打开高速下载网站:https://doget.nocsdn.com 第二步 将文件地址输入进去,然后回 ...
- react发布一个组件库 系列篇(一)
前言 经常使用别人写好的组件库,然后安装引入使用即可.比如: npm install beautiful-table import BeautifulTable from 'beautiful-tab ...
- JavaScript真的需要链表吗?
javaScript可以原生提供的数据类型的确有限,但是并不代表不需要. 从一开始只有Object.Array到现在增加的Map和Set也确实证明前端也在不断发展自己的数据结构. 下边就有些没有的数据 ...
- 前端开发系列087-Node篇之Buffer
一.Buffer介绍 Buffer是Node中特有的数据类型,它是Node作为运行时对JavaScript进行的拓展,专门用来处理二进制数据流.Buffer属于固有(built-in)类型的全局变量, ...
- FT 终面记录
可能挂了 写一个记录总结一下吧 终面 我很庆幸能走到终面 前面很多次我都很可能挂掉 前面面试不是很紧张就是一直觉得自己会挂掉 终面了想想自己很可能会进 FT 有点紧张了 吾 现在还做不到 宠辱 ...
- ovm 格式的 六面体网格 样例 正方体的生成程序和文件
简介 Hexahedron 六面体网格,基于 openVolumeMesh 格式的程序生成cube code // C++ includes #include <iostream> #in ...