2024-02-18 20:56:59 +08:00
|
|
|
|
#region
|
|
|
|
|
|
2023-09-24 15:59:43 +08:00
|
|
|
|
using Unity.Burst;
|
|
|
|
|
using Unity.Collections;
|
2024-03-03 16:34:48 +08:00
|
|
|
|
using Unity.Collections.LowLevel.Unsafe;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
using Unity.Jobs;
|
|
|
|
|
|
2024-02-18 20:56:59 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
2023-09-24 15:59:43 +08:00
|
|
|
|
namespace nadena.dev.modular_avatar.core.armature_lock
|
|
|
|
|
{
|
2024-03-03 16:34:48 +08:00
|
|
|
|
internal class BidirectionalArmatureLockOperator : ArmatureLockOperator<BidirectionalArmatureLockOperator>
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
private NativeArrayRef<TransformState> SavedState;
|
2024-03-03 16:34:48 +08:00
|
|
|
|
protected override bool WritesBaseBones => true;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
public BidirectionalArmatureLockOperator()
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
SavedState = _memoryManager.CreateArray<TransformState>();
|
|
|
|
|
}
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
protected override bool SetupJob(ISegment segment)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < segment.Length; i++)
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
int bone = i + segment.Offset;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
var baseBone = BaseTransforms[bone];
|
|
|
|
|
var targetBone = TargetTransforms[bone];
|
|
|
|
|
|
|
|
|
|
SavedState.Array[i] = TransformState.FromTransform(targetBone);
|
|
|
|
|
if (TransformState.Differs(TransformState.FromTransform(baseBone), SavedState.Array[i]))
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
return false;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-05 16:26:30 +08:00
|
|
|
|
|
|
|
|
|
return true;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
protected override JobHandle Compute(ArmatureLockJobAccessor accessor, int? jobIndex, JobHandle dependency)
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-03 16:34:48 +08:00
|
|
|
|
return new ComputeOperator()
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-03 16:34:48 +08:00
|
|
|
|
base_in = accessor._in_baseBone,
|
|
|
|
|
merge_in = accessor._in_targetBone,
|
|
|
|
|
base_out = accessor._out_baseBone,
|
|
|
|
|
merge_out = accessor._out_targetBone,
|
|
|
|
|
|
|
|
|
|
SavedState = SavedState,
|
|
|
|
|
baseDirty = accessor._out_dirty_baseBone,
|
|
|
|
|
mergeDirty = accessor._out_dirty_targetBone,
|
|
|
|
|
boneToJobIndex = accessor._boneToJobIndex,
|
|
|
|
|
wroteAny = accessor._didAnyWriteFlag,
|
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
boneInUse = accessor._in_boneInUse,
|
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
singleJobIndex = jobIndex ?? -1
|
|
|
|
|
}.Schedule(accessor._in_baseBone.Length, 16, dependency);
|
2023-09-24 15:59:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
[BurstCompile]
|
|
|
|
|
private struct ComputeOperator : IJobParallelFor
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-03 16:34:48 +08:00
|
|
|
|
public int singleJobIndex;
|
2024-02-26 17:03:16 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
public NativeArray<TransformState> base_in, merge_in, base_out, merge_out;
|
2024-02-26 17:03:16 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
public NativeArray<TransformState> SavedState;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
[WriteOnly] public NativeArray<bool> baseDirty, mergeDirty;
|
2024-03-03 16:34:48 +08:00
|
|
|
|
[ReadOnly] public NativeArray<int> boneToJobIndex;
|
2024-02-17 18:20:08 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
[NativeDisableContainerSafetyRestriction] [NativeDisableParallelForRestriction] [WriteOnly]
|
2024-03-05 16:26:30 +08:00
|
|
|
|
public NativeArray<bool> wroteAny;
|
|
|
|
|
|
|
|
|
|
[ReadOnly] public NativeArray<bool> boneInUse;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
[BurstCompile]
|
|
|
|
|
public void Execute(int index)
|
2023-09-24 15:59:43 +08:00
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
if (!boneInUse[index]) return;
|
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
var jobIndex = boneToJobIndex[index];
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
if (singleJobIndex != -1 && jobIndex != singleJobIndex) return;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
var baseBone = base_in[index];
|
|
|
|
|
var mergeBone = merge_in[index];
|
|
|
|
|
var saved = SavedState[index];
|
2024-02-26 17:03:16 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
if (TransformState.Differs(saved, mergeBone))
|
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
baseDirty[index] = true;
|
|
|
|
|
mergeDirty[index] = false;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
SavedState[index] = base_out[index] = merge_in[index];
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
wroteAny[jobIndex] = true;
|
2024-03-03 16:34:48 +08:00
|
|
|
|
}
|
|
|
|
|
else if (TransformState.Differs(saved, baseBone))
|
|
|
|
|
{
|
2024-03-05 16:26:30 +08:00
|
|
|
|
mergeDirty[index] = true;
|
|
|
|
|
baseDirty[index] = false;
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-03 16:34:48 +08:00
|
|
|
|
SavedState[index] = merge_out[index] = base_in[index];
|
2023-09-24 15:59:43 +08:00
|
|
|
|
|
2024-03-05 16:26:30 +08:00
|
|
|
|
wroteAny[jobIndex] = true;
|
2024-03-03 16:34:48 +08:00
|
|
|
|
}
|
2023-09-24 15:59:43 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|