194 lines
5.8 KiB
C#
194 lines
5.8 KiB
C#
using UnityEngine;
|
|
using KindRetargeting;
|
|
|
|
[DefaultExecutionOrder(16001)]
|
|
public class SimplePoseTransfer : MonoBehaviour
|
|
{
|
|
[Header("Pose Transfer Settings")]
|
|
public Animator sourceBone;
|
|
public Animator[] targetBones;
|
|
|
|
[Header("Scale Transfer")]
|
|
[Tooltip("소스의 머리 스케일을 타겟에도 적용")]
|
|
public bool transferHeadScale = true;
|
|
|
|
// 캐싱된 Transform들
|
|
private Transform[] cachedSourceBones;
|
|
private Transform[,] cachedTargetBones;
|
|
|
|
// 각 targetBone의 초기 회전 차이를 저장
|
|
private Quaternion[,] boneRotationDifferences;
|
|
|
|
// 머리 스케일 관련
|
|
private CustomRetargetingScript sourceRetargetingScript;
|
|
private Vector3[] originalTargetHeadScales;
|
|
|
|
private void Start()
|
|
{
|
|
Init();
|
|
}
|
|
|
|
public void Init()
|
|
{
|
|
if (targetBones == null || targetBones.Length == 0)
|
|
{
|
|
Debug.LogError("Target bones are null or empty");
|
|
return;
|
|
}
|
|
|
|
if (sourceBone == null)
|
|
{
|
|
Debug.LogError("Source bone is null");
|
|
return;
|
|
}
|
|
|
|
InitializeTargetBones();
|
|
CacheAllBoneTransforms();
|
|
CacheHeadScales();
|
|
|
|
Debug.Log($"SimplePoseTransfer initialized with {targetBones.Length} targets");
|
|
}
|
|
|
|
private void CacheHeadScales()
|
|
{
|
|
// 소스에서 CustomRetargetingScript 찾기
|
|
sourceRetargetingScript = sourceBone.GetComponent<CustomRetargetingScript>();
|
|
|
|
// 타겟들의 원본 머리 스케일 저장
|
|
originalTargetHeadScales = new Vector3[targetBones.Length];
|
|
for (int i = 0; i < targetBones.Length; i++)
|
|
{
|
|
if (targetBones[i] != null)
|
|
{
|
|
Transform headBone = targetBones[i].GetBoneTransform(HumanBodyBones.Head);
|
|
if (headBone != null)
|
|
{
|
|
originalTargetHeadScales[i] = headBone.localScale;
|
|
}
|
|
else
|
|
{
|
|
originalTargetHeadScales[i] = Vector3.one;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private void InitializeTargetBones()
|
|
{
|
|
boneRotationDifferences = new Quaternion[targetBones.Length, 55];
|
|
|
|
for (int i = 0; i < targetBones.Length; i++)
|
|
{
|
|
if (targetBones[i] == null)
|
|
{
|
|
Debug.LogError($"targetBones[{i}] is null");
|
|
continue;
|
|
}
|
|
|
|
// 55개의 휴머노이드 본에 대해 회전 차이 계산
|
|
for (int j = 0; j < 55; j++)
|
|
{
|
|
Transform sourceBoneTransform = sourceBone.GetBoneTransform((HumanBodyBones)j);
|
|
Transform targetBoneTransform = targetBones[i].GetBoneTransform((HumanBodyBones)j);
|
|
|
|
if (sourceBoneTransform != null && targetBoneTransform != null)
|
|
{
|
|
boneRotationDifferences[i, j] = Quaternion.Inverse(sourceBoneTransform.rotation) * targetBoneTransform.rotation;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private void CacheAllBoneTransforms()
|
|
{
|
|
// 소스 본 캐싱
|
|
cachedSourceBones = new Transform[55];
|
|
for (int i = 0; i < 55; i++)
|
|
{
|
|
cachedSourceBones[i] = sourceBone.GetBoneTransform((HumanBodyBones)i);
|
|
}
|
|
|
|
// 타겟 본 캐싱
|
|
cachedTargetBones = new Transform[targetBones.Length, 55];
|
|
for (int t = 0; t < targetBones.Length; t++)
|
|
{
|
|
for (int i = 0; i < 55; i++)
|
|
{
|
|
cachedTargetBones[t, i] = targetBones[t].GetBoneTransform((HumanBodyBones)i);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void LateUpdate()
|
|
{
|
|
TransferPoses();
|
|
}
|
|
|
|
private void TransferPoses()
|
|
{
|
|
if (sourceBone == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < targetBones.Length; i++)
|
|
{
|
|
if (targetBones[i] != null && targetBones[i].gameObject.activeInHierarchy)
|
|
{
|
|
TransferPoseToTarget(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void TransferPoseToTarget(int targetIndex)
|
|
{
|
|
Animator targetBone = targetBones[targetIndex];
|
|
|
|
// 루트 회전 동기화
|
|
targetBone.transform.rotation = sourceBone.transform.rotation;
|
|
|
|
// 모든 본에 대해 포즈 전송
|
|
for (int i = 0; i < 55; i++)
|
|
{
|
|
Transform targetBoneTransform = cachedTargetBones[targetIndex, i];
|
|
Transform sourceBoneTransform = cachedSourceBones[i];
|
|
|
|
if (targetBoneTransform != null && sourceBoneTransform != null)
|
|
{
|
|
// 회전 적용
|
|
targetBoneTransform.rotation = sourceBoneTransform.rotation * boneRotationDifferences[targetIndex, i];
|
|
|
|
// 펠비스 위치 동기화 (HumanBodyBones.Hips = 0)
|
|
if (i == 0)
|
|
{
|
|
targetBoneTransform.position = sourceBoneTransform.position;
|
|
}
|
|
|
|
// 머리 스케일 적용 (HumanBodyBones.Head = 10)
|
|
if (transferHeadScale && i == 10)
|
|
{
|
|
ApplyHeadScale(targetIndex, targetBoneTransform);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private void ApplyHeadScale(int targetIndex, Transform targetHeadBone)
|
|
{
|
|
if (sourceRetargetingScript != null)
|
|
{
|
|
float headScale = sourceRetargetingScript.GetHeadScale();
|
|
targetHeadBone.localScale = originalTargetHeadScales[targetIndex] * headScale;
|
|
}
|
|
else
|
|
{
|
|
// CustomRetargetingScript가 없으면 소스 머리 스케일 직접 복사
|
|
Transform sourceHead = cachedSourceBones[10]; // HumanBodyBones.Head
|
|
if (sourceHead != null)
|
|
{
|
|
targetHeadBone.localScale = sourceHead.localScale;
|
|
}
|
|
}
|
|
}
|
|
|
|
} |