Dots:动态实现GPUECSAnimationBaker的受击变红效果
前言
如下图所示,直播玩法中需要实现受击变红的效果:
关于大批量加载,可以参考我的上篇博客。
关于实现方式,网上参考资料较少,经过一段时间的研究,我终于实现了以上效果。
大致步骤为:
1,修改shader group,以实现对修改颜色的支持
2,在加载的实体上挂载脚本,以方便能找到修改的实体
3,代码动态更改IComponentData的值,以实现对物体的颜色控制
1,修改shader group
实现原理参考了这篇官方帖子。
关于shader group的基础知识这里不做赘述,直接拷贝原始模版:
并进行如下修改:
a,添加lerp
在原有basecolor中添加一个lerp,以实现将叠加的红色赋值进来
b,公开addValue属性
修改addValue的公开属性,为可更改的值
c,查看属性
最后,我们可以在引用的材质球上,看到addValue的属性
这就表明,我们可以在代码中去修改这个参数。
2,需要修改的实体挂载脚本
a,新建一个IComponentData
在dots里,每个属性基本都是对应一个IComponentData。
所以,我们声明一个Render_AddColor,其中[MaterialProperty("_addValue")]代表shader group中公开的值:
[BurstCompile][MaterialProperty("_addValue")]public struct Render_AddColor : IComponentData{public float value;public Render_AddColor(float c) { value = c; }}
我这里为了省事,把这个放到这块了:
这个放到别的地方应该不影响。
b,新建一个ChangeColorBarkerBehaviour脚本
这个脚本的作用在于,存储需要修改的实体:
using UnityEngine;
using Unity.Entities;namespace GPUECSAnimationBaker.Engine.AnimatorSystem
{public class ChangeColorBarkerBehaviour : MonoBehaviour{public GameObject[] prefabs;}public class ChangeColorBarker : Baker<ChangeColorBarkerBehaviour>{public override void Bake(ChangeColorBarkerBehaviour authoring){Entity entity = GetEntity(TransformUsageFlags.None);// 添加一个动态缓冲区来存储每个 Prefab 的实体DynamicBuffer<AddColorBufferElement> addColorBuffer = AddBuffer<AddColorBufferElement>(entity);// 遍历 prefabs 数组for (int prefabIndex = 0; prefabIndex < authoring.prefabs.Length; prefabIndex++){// 获取 Prefab 对应的实体Entity prefabEntity = GetEntity(authoring.prefabs[prefabIndex], TransformUsageFlags.Dynamic); // 将实体添加到缓冲区为实体添加 Render_AddColor 组件,这里有点问题//AddComponent(prefabEntity, new Render_AddColor()//{// value = 0//});addColorBuffer.Add(new AddColorBufferElement(){entity = prefabEntity});}}}// 定义一个缓冲区元素类型public struct AddColorBufferElement : IBufferElementData{public Entity entity;}
}
b,找到需要修改的实体
挂载方式如下:
这样,就可以动态获取到该实体。
3,动态修改需要改变的值
这一步其实就比较简单了,首先获取到entity实体,修改函数如下:
public void ChangeColor2(EntityManager entityManager, Entity e){DynamicBuffer<AddColorBufferElement> buffer = entityManager.GetBuffer<AddColorBufferElement>(e);Debug.Log("Buffer Length: " + buffer.Length);// 创建一个列表来存储缓冲区中的元素var list = new List<AddColorBufferElement>();foreach (var element in buffer){list.Add(element);}// 遍历列表并处理每个元素for (int i = list.Count - 1; i >= 0; i--){var bufferElement = list[i];Entity prefabEntity = bufferElement.entity;// 确保实体存在if (!entityManager.Exists(prefabEntity)){Debug.LogWarning($"Entity {prefabEntity} does not exist.");continue;}// 检查并添加 Render_AddColor 组件if (!entityManager.HasComponent<Render_AddColor>(prefabEntity)){Debug.Log("添加");entityManager.AddComponentData(prefabEntity, new Render_AddColor(0));}else{// 获取组件数据并更新其值Render_AddColor addColor = entityManager.GetComponentData<Render_AddColor>(prefabEntity);Debug.Log($"Entity: {prefabEntity}, AddColor Value: {addColor.value}");// 更新组件的值addColor.value = Random.Range(0.3f, 0.9f);entityManager.SetComponentData(prefabEntity, addColor);}}}
这个函数是测试用的, 作用在于每秒随机显示不同value的红色,最终测试画面如下:
4,结语
如果有其他的需求,可以自定义shadergroup,然后自定义公开值。
有需要可以联系我。