273 lines
8.6 KiB
C#
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;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|