Unity引擎UGUI上特效处理重叠和裁剪问题的多种解决办法

本文主要是介绍Unity引擎UGUI上特效处理重叠和裁剪问题的多种解决办法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  大家好,我是阿赵。
  使用Unity引擎开发项目,使用UGUI做界面,经常会遇到需要把特效放在UI上,但UI本身和特效又需要有遮挡关系和裁剪效果。
  之前我介绍了一下使用MaskableGraphic的方式把粒子特效渲染在UI上,把粒子特效当Image用。实际上解决类似的问题,是有很多手段的,这些手段各有优缺点,实现的难度也各有不同,这篇文章不会很详细的列举代码,但可以从思路上分享一下。

一、 RenderTexture

  这是最直观的方法,把特效放在一个远离主场景的地方,然后单独打一个摄像机,设置一个RenderTexture作为摄像机的TargetTexture。这样摄像机会把特效渲染到这张RenderTexture上面,然后我们就可以把这种RenderTexture用作RawImage的贴图,正常重叠和裁剪了。
  这个方法从操作上来说比较简单实现,但实际上有很多缺点:
1、 为每一个特效都打一个摄像机,看着就很傻
2、 生成多张RenderTexture,严重占用内存。而由于每个特效的大小都不一样,需要使用多大的RenderTexture也是一个问题。
3、 对于特效本身是半透明渲染的,在RenderTexture上透明通道会出现问题。因为比如是AlphaBlend这种叠加方式,实际上是需要把半透明颜色叠加在背景色才能显示出正常的效果。但由于我们单独打了一个摄像机看这个特效,并没有合理的背景色作为叠加,所以渲染出来的图片透明通道是不正确的。解决办法当然也有,先把摄像机背景色调成纯黑,渲染一张,再把摄像机背景色调成纯白,再渲染一张,把两张图通过计算可以得到正确的透明通道和颜色。不过这样做,感觉就更麻烦了。

二、 修改渲染的顺序

  这个方法我觉得可能是最多人用的。
  首先我们要知道,UGUI的Canvas上面有一个Order in Layer的参数。这个参数实际上是这一张Canvas在渲染时候的渲染顺序。
在这里插入图片描述

  一套UI系统里面,可以有多个Canvas层,order大的Canvas会盖住order小的Canvas。
  然后我们需要知道,在Mesh渲染器或者粒子渲染器里面,也有一个渲染order:
在这里插入图片描述

  这个Order in Layer和Canvas里面那个Order In Layer是同一回事。不过有个问题是,在Canvas里面的UI元素,比如一张Image,如果自身没有单独的Canvas,那么它的Order会和父级的Canvas一样。而同一个父级下的多个Image,排序是按照sibling index来排的,也就是作为子物体的顺序。但如果一个粒子特效和一个Image,他们的Order In Layer一样,却并不会按照sibling index来排序,粒子特效会盖住Image。
  知道了这个规则之后,问题就比较简单处理了。如果特效要叠加在UI上面,那么把特效的Order设置到和UI的父级Canvas一样就行。
  再举一个复杂点的例子,假如父级Canvas下面有一张Image1,然后有一个粒子特效fx要盖住Image1,然后还有一张Image2要盖住fx。比如父级Canvas的order是100,那么fx的order也设置成100,fx就自然盖住Image1了。接下来,在Image2身上加一个Canvas,把Override Sortting勾上,然后order填101,这是,Image就会把Image1和fx都盖住了。
  如果用代码实现,其实就很简单了,写一个C#脚本,挂在需要改变order的对象上,然后提供一个方法,设置一个添加order的数量。调用方法是,找自身的父级,一直找到父级有Canvas为止,得到父级Canvas的order,比如是100,然后加上传入参数,比如传入了1,那么得到的自己的order就是101了。然后先Get自身的Component,看看有没有renderer,如果有,则renderer.sortingOrder来设置renderer的order。如果没有renderer,则应该是UI元素,就给自己AddComponent一个Canvas,把order设置一下就行了。
  我之前说这个方法最多人用,是因为实现简单,也比较的合理,不过这个方法也不是完全没有缺点的。
1、 由于在Canvas下面的子物体又添加了Canvas,会导致原本可以合并渲染的UI元件变得不能合并渲染
2、 如果只是简单的叠一层特效到UI上,问题不大。但如果UI和特效互相叠加的层级多起来之后,维护order会变得复杂,比较容易出错。
3、 Order In Layer只会解决层叠问题,不会提供裁剪效果的,所以如果需要用Mask裁剪,还需要对特效使用的shader做处理,一般的处理方式是在特效上面挂一个C#脚本,不停的获取父级Mask的Rect,然后转换成世界坐标传入给特效使用的材质球的Shader,在Shader上面判断在一定世界坐标范围外的片元就舍弃掉,把Alpha变成0,再Clip掉。

三、 MaskableGraphic

  上一篇文章介绍了怎样在MaskableGraphic上面用UIVertex绘制顶点,模拟粒子的网格渲染。
  实际上MaskableGraphic可以理解成是一个可以渲染任意Mesh网格的图层,除了用UIVertex来逐个粒子计算和绘制,还可以通过BakeMesh,把特效的网格Bake出来,再传到MaskableGraphic里面渲染。
  由于方法太自由,反而很难具体的说明。可以当做Unity提供了一个绘制顶点和三角面的渲染API,自己想画什么就画什么。

四、 用其他方式替代粒子

  这个可以归结为其他方法了。核心思想就是不用粒子系统做特效。
  最常见的是用序列帧替代一些复杂的效果。不过序列帧不适合做面积较大还有细节太多的动画,不然由于帧数太多和单帧太大,会导致容量很大。
  另外一种常见的,是用Animation的方式通过K帧实现。结合部分序列帧,可以实现大部分效果。不过像大面积数量很多的粒子扩散效果,就比较难实现。
  还有比较常见的,是用Spine之类的2D骨骼动画来做特效。这类2D动画插件,一般除了控制骨骼运动以外,还会带有序列帧和流光等常用的功能,可以做出很多动画效果。而Spine是有Unity的UGUI接口的,所以可以正常当Image使用。
  或者是用Shader直接做各种动画效果。这个就比较多样化,通过在Image上面挂一个材质球,然后通过Shader实现过载效果,常用的比如UV动画、用噪声图或者法线图模拟扭曲或者叠加的效果,用黑白模拟闪烁效果,等等,完全可以发挥自己的想象力。包括大量粒子扩散的效果,我也用Shader模拟过。
  以上的这些方法,由于没有使用UI以外的渲染方式,层叠和裁剪一般都不会有问题。但由于不同的效果需要使用的方法不一样,甚至是一个特效要K好多动画和写好几种shader,所以一般的特效师是很难做到的。不过我自己的确是这样做特效的,所以也不妨提出来,算是一种解决的办法。

这篇关于Unity引擎UGUI上特效处理重叠和裁剪问题的多种解决办法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072398

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

轻量级在线服装3D定制引擎Myway简介

我写的面向web元宇宙轻量级系列引擎中的另外一个,在线3D定制引擎Myway 3D。 用于在线商品定制,比如个性化服装的定制、日常用品(如杯子)、家装(被套)等物品的在线定制。 特性列表: 可更换衣服款式,按需定制更换模型可实时更改材质颜色可实时添加文本,并可实时修改大小、颜色和角度,支持自定义字体可实时添加艺术图标,并可实时修改大小、颜色和角度,支持翻转、各种对齐可更改衣服图案,按需求定制

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

2024.6.24 IDEA中文乱码问题(服务器 控制台 TOMcat)实测已解决

1.问题产生原因: 1.文件编码不一致:如果文件的编码方式与IDEA设置的编码方式不一致,就会产生乱码。确保文件和IDEA使用相同的编码,通常是UTF-8。2.IDEA设置问题:检查IDEA的全局编码设置和项目编码设置是否正确。3.终端或控制台编码问题:如果你在终端或控制台看到乱码,可能是终端的编码设置问题。确保终端使用的是支持你的文件的编码方式。 2.解决方案: 1.File -> S

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

问题-windows-VPN不正确关闭导致网页打不开

为什么会发生这类事情呢? 主要原因是关机之前vpn没有关掉导致的。 至于为什么没关掉vpn会导致网页打不开,我猜测是因为vpn建立的链接没被更改。 正确关掉vpn的时候,会把ip链接断掉,如果你不正确关掉,ip链接没有断掉,此时你vpn又是没启动的,没有域名解析,所以就打不开网站。 你可以在打不开网页的时候,把vpn打开,你会发现网络又可以登录了。 方法一 注意:方法一虽然方便,但是可能会有

(超详细)YOLOV7改进-Soft-NMS(支持多种IoU变种选择)

1.在until/general.py文件最后加上下面代码 2.在general.py里面找到这代码,修改这两个地方 3.之后直接运行即可

iOS HTTPS证书不受信任解决办法

之前开发App的时候服务端使用的是自签名的证书,导致iOS开发过程中调用HTTPS接口时,证书不被信任 - (void)URLSession:(NSURLSession *)session didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge completionHandler:(void (^)(NSURLSessionAu

亮相WOT全球技术创新大会,揭秘火山引擎边缘容器技术在泛CDN场景的应用与实践

2024年6月21日-22日,51CTO“WOT全球技术创新大会2024”在北京举办。火山引擎边缘计算架构师李志明受邀参与,以“边缘容器技术在泛CDN场景的应用和实践”为主题,与多位行业资深专家,共同探讨泛CDN行业技术架构以及云原生与边缘计算的发展和展望。 火山引擎边缘计算架构师李志明表示:为更好地解决传统泛CDN类业务运行中的问题,火山引擎边缘容器团队参考行业做法,结合实践经验,打造火山

vue同页面多路由懒加载-及可能存在问题的解决方式

先上图,再解释 图一是多路由页面,图二是路由文件。从图一可以看出每个router-view对应的name都不一样。从图二可以看出层路由对应的组件加载方式要跟图一中的name相对应,并且图二的路由层在跟图一对应的页面中要加上components层,多一个s结尾,里面的的方法名就是图一路由的name值,里面还可以照样用懒加载的方式。 页面上其他的路由在路由文件中也跟图二是一样的写法。 附送可能存在