1
0
mirror of synced 2025-01-19 09:17:30 +01:00

Fix GFBANM rotation interpolation

This commit is contained in:
HelloOO7 2020-10-19 17:09:25 +02:00 committed by GitHub
parent 8dac30200a
commit 5d9981e6dc
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -187,6 +187,7 @@ namespace FirstPlugin
var node = animGroup as BoneGroup;
STBone b = null;
b = skeleton.GetBone(node.Name);
if (b == null) continue;
@ -211,11 +212,32 @@ namespace FirstPlugin
if (node.RotationX.HasKeys || node.RotationY.HasKeys || node.RotationZ.HasKeys)
{
short value1 = (short)node.RotationX.GetFrameValue(Frame);
short value2 = (short)node.RotationY.GetFrameValue(Frame);
short value3 = (short)node.RotationZ.GetFrameValue(Frame);
STKeyFrame xL = GetFVGFL(node.RotationX, Frame);
STKeyFrame yL = GetFVGFL(node.RotationY, Frame);
STKeyFrame zL = GetFVGFL(node.RotationZ, Frame);
STKeyFrame xR = GetFVGFR(node.RotationX, Frame);
STKeyFrame yR = GetFVGFR(node.RotationY, Frame);
STKeyFrame zR = GetFVGFR(node.RotationZ, Frame);
b.rot = PackedToQuat(value1, value2, value3);
short value1 = (short)xL.Value;
short value2 = (short)yL.Value;
short value3 = (short)zL.Value;
Quaternion rotL = PackedToQuat(value1, value2, value3);
short value1r = (short)xR.Value;
short value2r = (short)yR.Value;
short value3r = (short)zR.Value;
Quaternion rotR = PackedToQuat(value1r, value2r, value3r);
float weight = (Frame - xL.Frame) / (xR.Frame - xL.Frame);
if (xR.Frame - xL.Frame == 0) //NaN if divided by zero
{
weight = 0;
}
b.rot = Quaternion.Slerp(rotL, rotR, weight);
}
else
{
@ -228,10 +250,44 @@ namespace FirstPlugin
{
skeleton.update();
}
}
public STKeyFrame GetFVGFL(STAnimationTrack t, float frame, float startFrame = 0)
{
if (t.KeyFrames.Count == 0) return new STKeyFrame(frame, 0);
if (t.KeyFrames.Count == 1) return t.KeyFrames[0];
STKeyFrame LK = t.KeyFrames.First();
float Frame = frame - startFrame;
foreach (STKeyFrame keyFrame in t.KeyFrames)
{
if (keyFrame.Frame <= Frame) LK = keyFrame;
}
return LK;
}
// private static readonly ushort _flagsMask = 0b11000011_11111111;
public STKeyFrame GetFVGFR(STAnimationTrack t, float frame, float startFrame = 0)
{
if (t.KeyFrames.Count == 0) return new STKeyFrame(frame, 0);
if (t.KeyFrames.Count == 1) return t.KeyFrames[0];
STKeyFrame RK = t.KeyFrames.Last();
float Frame = frame - startFrame;
foreach (STKeyFrame keyFrame in t.KeyFrames)
{
if (keyFrame.Frame >= Frame && keyFrame.Frame < RK.Frame) RK = keyFrame;
}
return RK;
}
// private static readonly ushort _flagsMask = 0b11000011_11111111;
private static short UnpackS15(short u15)
{
int sign = (u15 >> 14) & 1;