Rasagar/Library/PackageCache/com.unity.visualeffectgraph/Editor/Compiler/VFXExpressionMapper.cs

155 lines
4.7 KiB
C#
Raw Normal View History

2024-08-26 13:07:20 -07:00
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Linq;
using UnityEngine.VFX;
namespace UnityEditor.VFX
{
struct VFXNamedExpression
{
public VFXNamedExpression(VFXExpression exp, string name)
{
this.exp = exp;
this.name = name;
}
public VFXExpression exp;
public string name;
}
class VFXExpressionMapper
{
public struct Data
{
public string fullName { get { return id == -1 ? name : string.Format("{0}_{1}", name, VFXCodeGeneratorHelper.GeneratePrefix((uint)id)); } }
public string name;
public int id;
}
public VFXExpressionMapper()
{
}
public IEnumerable<VFXExpression> expressions => m_ExpressionsData.Keys;
public void AddExpressionsFromSlotContainer(IVFXSlotContainer slotContainer, int blockId)
{
if (slotContainer.activationSlot != null)
AddExpressionsFromSlot(slotContainer.activationSlot, blockId);
foreach (var master in slotContainer.inputSlots)
AddExpressionsFromSlot(master, blockId);
}
public void AddExpressionsFromSlot(VFXSlot masterSlot, int blockId)
{
foreach (var slot in masterSlot.GetExpressionSlots())
{
var exp = slot.GetExpression();
if (!Contains(exp))
AddExpression(exp, slot.fullName, blockId);
}
}
public static VFXExpressionMapper FromBlocks(IEnumerable<VFXBlock> blocks)
{
var mapper = new VFXExpressionMapper();
int cpt = 0;
foreach (var block in blocks)
{
mapper.AddExpression(block.activationExpression, VFXBlock.activationSlotName, cpt);
mapper.AddExpressions(block.parameters, cpt++);
}
return mapper;
}
public static VFXExpressionMapper FromContext(VFXContext context)
{
var mapper = FromBlocks(context.activeFlattenedChildrenWithImplicit);
mapper.AddExpressionsFromSlotContainer(context, -1);
return mapper;
}
public ReadOnlyCollection<Data> GetData(VFXExpression exp)
{
List<Data> data;
if (m_ExpressionsData.TryGetValue(exp, out data))
{
return data.AsReadOnly();
}
return (new List<Data>()).AsReadOnly();
}
public bool Contains(VFXExpression exp)
{
return m_ExpressionsData.ContainsKey(exp);
}
public void AddExpression(VFXExpression exp, Data data)
{
AddExpression(exp, data.name, data.id);
}
public VFXExpression FromNameAndId(string name, int id)
{
foreach (var expression in m_ExpressionsData)
{
if (expression.Value.Any(o => o.name == name && o.id == id))
{
return expression.Key;
}
}
return null;
}
public IEnumerable<VFXNamedExpression> CollectExpression(int id, bool fullname = true)
{
foreach (var expressionData in m_ExpressionsData)
{
foreach (var data in expressionData.Value)
{
if (data.id == id)
{
yield return new VFXNamedExpression(expressionData.Key, fullname ? data.fullName : data.name);
}
}
}
}
public void AddExpression(VFXExpression exp, string name, int id)
{
if (exp == null || name == null)
throw new ArgumentNullException();
if (m_ExpressionsData.SelectMany(o => o.Value).Any(o => o.name == name && o.id == id))
throw new ArgumentException($"{name}_{id} has been added twice: {exp}");
var data = new Data();
data.name = name;
data.id = id;
if (!m_ExpressionsData.ContainsKey(exp))
{
m_ExpressionsData.Add(exp, new List<Data>() { data });
}
else
{
m_ExpressionsData[exp].Add(data);
}
}
public void AddExpression(VFXNamedExpression expression, int id)
{
AddExpression(expression.exp, expression.name, id);
}
public void AddExpressions(IEnumerable<VFXNamedExpression> expressions, int id)
{
foreach (var exp in expressions)
AddExpression(exp, id);
}
private readonly Dictionary<VFXExpression, List<Data>> m_ExpressionsData = new();
}
}