wuxianshengcong/Library/PackageCache/com.unity.2d.animation@9.1.2/Runtime/TransformAccessJob.cs
2025-01-02 14:50:41 +08:00

273 lines
8.6 KiB
C#

using System;
using System.Collections.Generic;
using Unity.Collections;
using Unity.Jobs;
using Unity.Mathematics;
using UnityEngine.Jobs;
using UnityEngine.Profiling;
using Unity.Burst;
namespace UnityEngine.U2D.Animation
{
internal class TransformAccessJob
{
internal struct TransformData
{
public int transformIndex;
public int refCount;
public TransformData(int index)
{
transformIndex = index;
refCount = 1;
}
}
Transform[] m_Transform;
TransformAccessArray m_TransformAccessArray;
NativeHashMap<int, TransformData> m_TransformData;
NativeArray<float4x4> m_TransformMatrix;
bool m_Dirty;
JobHandle m_JobHandle;
public TransformAccessJob()
{
m_TransformMatrix = new NativeArray<float4x4>(1, Allocator.Persistent);
m_TransformData = new NativeHashMap<int, TransformData>(1, Allocator.Persistent);
m_Transform = new Transform[0];
m_Dirty = false;
m_JobHandle = default(JobHandle);
}
public void Destroy()
{
m_JobHandle.Complete();
if (m_TransformMatrix.IsCreated)
m_TransformMatrix.Dispose();
if (m_TransformAccessArray.isCreated)
m_TransformAccessArray.Dispose();
if (m_TransformData.IsCreated)
m_TransformData.Dispose();
}
public NativeHashMap<int, TransformData> transformData
{
get { return m_TransformData; }
}
public NativeArray<float4x4> transformMatrix
{
get { return m_TransformMatrix; }
}
public void AddTransform(Transform t)
{
if (t == null || !m_TransformData.IsCreated)
return;
m_JobHandle.Complete();
int instanceId = t.GetInstanceID();
if (m_TransformData.ContainsKey(instanceId))
{
var transformData = m_TransformData[instanceId];
transformData.refCount += 1;
m_TransformData[instanceId] = transformData;
}
else
{
m_TransformData.TryAdd(instanceId, new TransformData(-1));
ArrayAdd(ref m_Transform, t);
m_Dirty = true;
}
}
static void ArrayAdd<T>(ref T[] array, T item)
{
int arraySize = array.Length;
Array.Resize(ref array, arraySize + 1);
array[arraySize] = item;
}
static void ArrayRemoveAt<T>(ref T[] array, int index)
{
List<T> list = new List<T>(array);
list.RemoveAt(index);
array = list.ToArray();
}
void UpdateTransformIndex()
{
if (!m_Dirty)
return;
m_Dirty = false;
Profiler.BeginSample("UpdateTransformIndex");
NativeArrayHelpers.ResizeIfNeeded(ref m_TransformMatrix, m_Transform.Length);
if (!m_TransformAccessArray.isCreated)
TransformAccessArray.Allocate(m_Transform.Length, -1, out m_TransformAccessArray);
else if (m_TransformAccessArray.capacity != m_Transform.Length)
m_TransformAccessArray.capacity = m_Transform.Length;
m_TransformAccessArray.SetTransforms(m_Transform);
for (int i = 0; i < m_Transform.Length; ++i)
{
if (m_Transform[i] != null)
{
var instanceId = m_Transform[i].GetInstanceID();
var transformData = m_TransformData[instanceId];
transformData.transformIndex = i;
m_TransformData[instanceId] = transformData;
}
}
Profiler.EndSample();
}
public JobHandle StartLocalToWorldJob()
{
if (m_Transform.Length > 0)
{
m_JobHandle.Complete();
UpdateTransformIndex();
Profiler.BeginSample("StartLocalToWorldJob");
var job = new LocalToWorldTransformAccessJob()
{
outMatrix = transformMatrix,
};
m_JobHandle = job.Schedule(m_TransformAccessArray);
Profiler.EndSample();
return m_JobHandle;
}
return default(JobHandle);
}
public JobHandle StartWorldToLocalJob()
{
if (m_Transform.Length > 0)
{
m_JobHandle.Complete();
UpdateTransformIndex();
Profiler.BeginSample("StartWorldToLocalJob");
var job = new WorldToLocalTransformAccessJob()
{
outMatrix = transformMatrix,
};
m_JobHandle = job.Schedule(m_TransformAccessArray);
Profiler.EndSample();
return m_JobHandle;
}
return default(JobHandle);
}
internal string GetDebugLog()
{
var log = "";
#if COLLECTIONS_2_0_OR_ABOVE
log += "TransformData Count: " + m_TransformData.Count + "\n";
#else
log += "TransformData Count: " + m_TransformData.Count() + "\n";
#endif
log += "Transform Count: " + m_Transform.Length + "\n";
foreach (var ss in m_Transform)
{
log += ss == null ? "null" : ss.name + " " + ss.GetInstanceID();
log += "\n";
if (ss != null)
{
log += "RefCount: " + m_TransformData[ss.GetInstanceID()].refCount + "\n";
}
log += "\n";
}
return log;
}
internal void RemoveTransformsByIds(IList<int> idsToRemove)
{
if (!m_TransformData.IsCreated)
return;
m_JobHandle.Complete();
for (var i = idsToRemove.Count - 1; i >= 0; --i)
{
var id = idsToRemove[i];
if (!m_TransformData.ContainsKey(id))
{
idsToRemove.Remove(id);
continue;
}
var transformData = m_TransformData[id];
if (transformData.refCount > 1)
{
transformData.refCount -= 1;
m_TransformData[id] = transformData;
idsToRemove.Remove(id);
}
}
if (idsToRemove.Count == 0)
return;
var transformList = new List<Transform>(m_Transform);
foreach (var id in idsToRemove)
{
m_TransformData.Remove(id);
var index = transformList.FindIndex(t => t.GetInstanceID() == id);
if (index >= 0)
transformList.RemoveAt(index);
}
m_Transform = transformList.ToArray();
}
internal void RemoveTransformById(int transformId)
{
if (!m_TransformData.IsCreated)
return;
m_JobHandle.Complete();
if (m_TransformData.ContainsKey(transformId))
{
var transformData = m_TransformData[transformId];
if (transformData.refCount == 1)
{
m_TransformData.Remove(transformId);
var index = Array.FindIndex(m_Transform, t => t.GetInstanceID() == transformId);
if (index >= 0)
{
ArrayRemoveAt(ref m_Transform, index);
}
m_Dirty = true;
}
else
{
transformData.refCount -= 1;
m_TransformData[transformId] = transformData;
}
}
}
}
[BurstCompile]
internal struct LocalToWorldTransformAccessJob : IJobParallelForTransform
{
[WriteOnly]
public NativeArray<float4x4> outMatrix;
public void Execute(int index, TransformAccess transform)
{
outMatrix[index] = transform.localToWorldMatrix;
}
}
[BurstCompile]
internal struct WorldToLocalTransformAccessJob : IJobParallelForTransform
{
[WriteOnly]
public NativeArray<float4x4> outMatrix;
public void Execute(int index, TransformAccess transform)
{
outMatrix[index] = transform.worldToLocalMatrix;
}
}
}