KINDNICK_URP/Assets/External/VRM10/Runtime/Components/LookAt/LookAtEyeDirectionApplicableToBone.cs
2025-04-25 21:14:54 +09:00

91 lines
3.3 KiB
C#

using System.Collections.Generic;
using UnityEngine;
namespace UniVRM10
{
/// <summary>
/// Apply eye direction to bone transforms.
/// </summary>
internal sealed class LookAtEyeDirectionApplicableToBone : ILookAtEyeDirectionApplicable
{
private readonly Transform _leftEye;
private readonly Quaternion _leftEyePreMultiplyRotation;
private readonly Quaternion _leftEyePostMultiplyRotation;
private readonly Transform _rightEye;
private readonly Quaternion _rightEyePreMultiplyRotation;
private readonly Quaternion _rightEyePostMultiplyRotation;
private readonly CurveMapper _horizontalOuter;
private readonly CurveMapper _horizontalInner;
private readonly CurveMapper _verticalDown;
private readonly CurveMapper _verticalUp;
public LookAtEyeDirectionApplicableToBone(Transform leftEye, Transform rightEye,
CurveMapper horizontalOuter, CurveMapper horizontalInner, CurveMapper verticalDown, CurveMapper verticalUp)
{
_leftEye = leftEye;
var leftEyeRotation = _leftEye.rotation;
_leftEyePreMultiplyRotation = _leftEye.localRotation * Quaternion.Inverse(leftEyeRotation);
_leftEyePostMultiplyRotation = leftEyeRotation;
_rightEye = rightEye;
var rightEyeRotation = _rightEye.rotation;
_rightEyePreMultiplyRotation = _rightEye.localRotation * Quaternion.Inverse(rightEyeRotation);
_rightEyePostMultiplyRotation = rightEyeRotation;
_horizontalOuter = horizontalOuter;
_horizontalInner = horizontalInner;
_verticalDown = verticalDown;
_verticalUp = verticalUp;
}
public void Apply(LookAtEyeDirection eyeDirection, Dictionary<ExpressionKey, float> actualWeights)
{
SetYawPitchToBones(eyeDirection);
}
public void Restore()
{
SetYawPitchToBones(new LookAtEyeDirection(0, 0));
}
private void SetYawPitchToBones(LookAtEyeDirection eyeDirection)
{
float leftYaw, rightYaw;
if (eyeDirection.Yaw < 0)
{
leftYaw = -_horizontalOuter.Map(-eyeDirection.Yaw);
rightYaw = -_horizontalInner.Map(-eyeDirection.Yaw);
}
else
{
leftYaw = _horizontalInner.Map(eyeDirection.Yaw);
rightYaw = _horizontalOuter.Map(eyeDirection.Yaw);
}
float pitch;
if (eyeDirection.Pitch < 0)
{
pitch = _verticalDown.Map(-eyeDirection.Pitch);
}
else
{
pitch = -_verticalUp.Map(eyeDirection.Pitch);
}
if (_leftEye != null)
{
_leftEye.localRotation = _leftEyePreMultiplyRotation *
Quaternion.Euler(pitch, leftYaw, 0) *
_leftEyePostMultiplyRotation;
}
if (_rightEye != null)
{
_rightEye.localRotation = _rightEyePreMultiplyRotation *
Quaternion.Euler(pitch, rightYaw, 0) *
_rightEyePostMultiplyRotation;
}
}
}
}