Files
Colosseum/Assets/External/Tools/SyntyPropBoneTool/Runtime/PropBoneConfig.cs

116 lines
5.8 KiB
C#

// Copyright (c) 2024 Synty Studios Limited. All rights reserved.
//
// Use of this software is subject to the terms and conditions of the Synty Studios End User Licence Agreement (EULA)
// available at: https://syntystore.com/pages/end-user-licence-agreement
//
// For additional details, see the LICENSE.MD file bundled with this software.
using UnityEngine;
namespace Synty.Tools.SyntyPropBoneTool
{
/// <summary>
/// A configuration class used to define what bones are called, where they should be in the hierarchy and what offsets to use when updating them at runtime.
/// </summary>
[CreateAssetMenu(menuName = "Synty/Animation/Synty Prop Bone Config", order = 1)]
public class PropBoneConfig : ScriptableObject
{
[Tooltip("The rig used to create the animations. Rig must be in T Pose.")]
public GameObject sourceRig;
[Tooltip("The rig to play the animations on in Unity. Rig must be in T Pose.")]
public GameObject targetRig;
[Tooltip("Parameters that define where prop bones will generate and how to constrain them")]
public PropBoneDefinition[] propBoneDefinitions;
/// <summary>
/// Attempts to calculate the rotation and scale offset for all the bone definitions based on sourceRig and targetRig.
/// For the offsets to calculate correctly the sourceRig and targetRig must both be in T Pose.
/// </summary>
[ContextMenu("Calculate Offset Values")]
public void CalculateOffsetValues()
{
if (sourceRig == null)
{
Debug.LogError($"Source rig is null in config {name}.", this);
}
if (targetRig == null)
{
Debug.LogError($"Target rig is null in config {name}.", this);
}
if (targetRig == null || sourceRig == null)
{
return;
}
for (int i = 0; i < propBoneDefinitions.Length; ++i)
{
// Check the bones in the definition exist in the source and target rigs
Transform sourceParent = TransformUtil.SearchHierarchy(sourceRig.transform, propBoneDefinitions[i].parentBoneName);
Transform targetParent = TransformUtil.SearchHierarchy(targetRig.transform, propBoneDefinitions[i].parentBoneName);
if (sourceParent == null)
{
Debug.LogError($"Cannot find bone in source rig called {propBoneDefinitions[i].parentBoneName}.");
continue;
}
if (targetParent == null)
{
Debug.LogError($"Cannot find bone in target rig called {propBoneDefinitions[i].parentBoneName}.");
continue;
}
// Calculate the rotational offset between the source and target rigs
propBoneDefinitions[i].rotationOffset = (Quaternion.Inverse(targetParent.rotation) * sourceParent.rotation).eulerAngles;
// Check the scale calculation bones are defined and exist in the source and target rigs
if (propBoneDefinitions[i].scaleCalculationBone1 == "" || propBoneDefinitions[i].scaleCalculationBone2 == "")
{
continue;
}
Transform sourceScaleBone1 = TransformUtil.SearchHierarchy(sourceRig.transform, propBoneDefinitions[i].scaleCalculationBone1);
Transform sourceScaleBone2 = TransformUtil.SearchHierarchy(sourceRig.transform, propBoneDefinitions[i].scaleCalculationBone2);
Transform targetScaleBone1 = TransformUtil.SearchHierarchy(targetRig.transform, propBoneDefinitions[i].scaleCalculationBone1);
Transform targetScaleBone2 = TransformUtil.SearchHierarchy(targetRig.transform, propBoneDefinitions[i].scaleCalculationBone2);
if (sourceScaleBone1 == null)
{
Debug.LogError($"Cannot find bone in source rig called {propBoneDefinitions[i].scaleCalculationBone1}.");
continue;
}
if (sourceScaleBone2 == null)
{
Debug.LogError($"Cannot find bone in source rig called {propBoneDefinitions[i].scaleCalculationBone2}.");
continue;
}
if (targetScaleBone1 == null)
{
Debug.LogError($"Cannot find bone in target rig called {propBoneDefinitions[i].scaleCalculationBone1}.");
continue;
}
if (targetScaleBone2 == null)
{
Debug.LogError($"Cannot find bone in target rig called {propBoneDefinitions[i].scaleCalculationBone2}.");
continue;
}
// Calculate the scale offset between the source and target rigs
propBoneDefinitions[i].scale = 1;
float sourceLength = Vector3.Distance(sourceScaleBone1.position, sourceScaleBone2.position);
float targetLength = Vector3.Distance(targetScaleBone1.position, targetScaleBone2.position);
if (sourceLength < float.Epsilon)
{
Debug.LogError($"Distance betweem source rig's scale bones is zero: scale bone 1: {propBoneDefinitions[i].scaleCalculationBone1}, scale bone 2 {propBoneDefinitions[i].scaleCalculationBone2}.");
continue;
}
if (targetLength < float.Epsilon)
{
Debug.LogError($"Distance betweem target rig's scale bones is zero: scale bone 1: {propBoneDefinitions[i].scaleCalculationBone1}, scale bone 2 {propBoneDefinitions[i].scaleCalculationBone2}.");
continue;
}
propBoneDefinitions[i].scale = targetLength / sourceLength;
}
}
}
}