Unity学习笔记(8) Unity残影效果整理,3D、2D、通用方案

Unity做残影效果是个不难的工作。
但是一些细节会对使用方法产生限制,这里整理一下各种情况下推荐的残影效果实现方案
1.3D网格 (高性能)
2.2D人物 (非骨骼动画) (高性能)
3.其它场景 (包括2D骨骼动画人物) (低性能)

3D人物

Unity3D的渲染主要是靠Mesh+Shader,
3D场景下人物往往用SkinnedMeshRenderer做网格渲染,这样方便换装系统。
无脑推荐这篇,针对使用了SkinnedMeshRenderer的网格(也就是带蒙皮的网格),
无需创建新实例,而是直接调用DrawMesh,性能更好使用也简单。
https://blog.csdn.net/qq992817263/article/details/52994907

2D人物 (非骨骼动画)

之所以不能无脑用上面的,是因为Unity中的2D图像渲染使用的不是Mesh而是Sprite,
用的渲染器也不是SkinnedMeshRenderer而是SpriteRenderer
因此原理上需要换一种方式。

也就是直接获取SpriteRenderersprite参数,然后作为残影绘制。
(也就是把2D精灵的图片单独拿出来画一下。)

网上有很多种做法,比较好的应该是这个朋友做的用粒子效果实现的:
https://www.bilibili.com/video/BV12V41117jW

其它场景 (包括2D骨骼动画人物)

如果上面的做法都不行,那就只能直接依靠最笨的方法,
在需要生成残影的位置克隆实体实现残影了。
如果有其它更高性能的方法,求大佬分享呜呜

比如使用2D骨骼做出来的精灵动画,
如果直接获取SpriteRenderersprite
你会发现根本和动画中的姿势不同,
而是默认姿势,像这样:
20220126215813
因为是依靠2D animation库中的spriteSkin脚本,
直接绘制到世界空间,而不是通过SpriteRenderer进行绘制。

又比如想要体现出换装效果,就不能静态设置指定的冲刺sprite

下面这个代码就是我自己写,也不复杂,基本上就是最低效的复制。
就是当需要生成残影时startSpawn=true时,
就计时间,对应间隔生成一些残影。
代码里设置生成的渲染层级是"PerGround",免得残影把本体挡住了,
用的话记得自己换成自己的背景层名。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class AfterImageManager : MonoBehaviour
{
    public GameObject go_BoneRoot;  //残影克隆对象
    public Transform tf_SpawnPosition; //残影产生位置
    public Color cr_AfterImageColor = Color.black; //颜色
    public float maxAfterTime = 1f; //产生残影持续时间
    public float maxIntervalSpawnTime = 0.2f; //产生残影间隔时间
    public float maxLiveTime = 0.5f;//残影生存时间
    public bool startSpawn = false;//是否开始生成
    float duringAfterTime = 0;
    float duringIntervalSpawnTime = 0;
    List<AfterImage> listAfterImage;//存放所有产生了的残影,方便管理与删除。
    private void Start() {
        listAfterImage = new List<AfterImage>();
    }
    void FixedUpdate() {
        if(startSpawn){
            if (duringAfterTime < maxAfterTime){
                duringAfterTime += Time.deltaTime;
                if(duringIntervalSpawnTime < maxIntervalSpawnTime){
                    duringIntervalSpawnTime += Time.deltaTime;
                }else{
                    duringIntervalSpawnTime = 0f;
                    spawnOne();
                }
            }else{
                startSpawn = false;
                duringAfterTime = 0f;
                duringIntervalSpawnTime = 0f;
            }
        }    
        for (int i = 0; i < listAfterImage.Count; i++){
            listAfterImage[i].update(Time.deltaTime);
        }
        while(listAfterImage.Count >0 && listAfterImage[0].isDestroyed){
            listAfterImage.RemoveAt(0);
        }
    }
    void spawnOne(){
        //listAfterImage.Add(new AfterImage(go_BoneRoot, tf_SpawnPosition, cr_AfterImageColor, (duringAfterTime/maxAfterTime)*maxLiveTime));
        listAfterImage.Add(new AfterImage(go_BoneRoot, tf_SpawnPosition, cr_AfterImageColor, maxLiveTime));
    }
}

class AfterImage{
    GameObject go;
    float maxLiveTime;
    float duringLiveTime = 0f;
    public bool isDestroyed = false;
    SpriteRenderer[] srList;
    public AfterImage(GameObject _go, Transform pos, Color cr, float maxLiveTime){
        this.go = GameObject.Instantiate(_go, pos.transform.position, pos.transform.rotation);
        this.go.transform.localScale = new Vector3(pos.localScale.x, pos.localScale.y, pos.localScale.z);
        this.maxLiveTime = maxLiveTime;
        srList =  this.go.GetComponentsInChildren<SpriteRenderer>();
        foreach(SpriteRenderer sr in srList){
            sr.color = cr;
            sr.sortingLayerName = "PerGround";
        }
    }
    public void update(float deltaTime){ //更新生存时间与透明化
        if(duringLiveTime < maxLiveTime){
            duringLiveTime += deltaTime;
            float alpha = 1f - duringLiveTime/maxLiveTime;
            foreach(SpriteRenderer sr in srList){
                sr.color = new Color(sr.color.r, sr.color.g, sr.color.b, alpha);
            }
        }else{
            GameObject.Destroy(go);
            isDestroyed = true;
        }
    }
}

20220126224626

如果直接用,复制的物体不能是被控制的实体,否则残影会和你一起走动,
或者可以自己在代码里自己将对应的脚本disable了。

  • 4
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Unity Spine 2D残影效果是一种在游戏开发中常用的特效技术,可以用来实现角色或物体移动时留下残影效果。通过在Spine动画中添加额外的骨骼或Sprite Renderer,并结合透明度和延迟淡出效果,可以实现残影特效的效果。 首先,在Spine中创建一个新的骨骼或Sprite Renderer,作为残影的载体。这个载体需要复制主角或物体的动画,可以通过复制骨骼及其关键帧并调整帧间延迟来实现。确保在复制动画的过程中,将残影骨骼或Sprite Renderer的透明度逐渐降低,以获得残影淡出效果。 接下来,在Unity中创建一个空物体作为残影对象的父物体,并将其位置与主角或物体保持一致。将残影载体作为父物体的子物体,并将残影载体的位置设置为与父物体的位置一致。 然后,通过控制残影骨骼的播放速度,以及调整透明度和延迟淡出效果,可以实现残影效果。可以尝试使用Animator组件或编写脚本来控制残影骨骼的动画播放速度和透明度变化。 最后,在游戏运行时,每一帧都需要更新残影对象的位置与主角或物体的位置保持一致,这样才能实现残影效果随着主角或物体的移动而产生。 总结来说,Unity Spine 2D残影效果的实现步骤主要包括创建残影载体,复制动画、调整透明度和延迟淡出效果,创建父物体并保持位置一致,控制残影载体的动画播放速度和透明度变化,以及更新残影对象的位置。通过这些步骤,可以实现一个逼真的残影效果,增强游戏的视觉效果

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值