-
Notifications
You must be signed in to change notification settings - Fork 200
/
Copy pathIKSwivelHingeJoint.cs
102 lines (86 loc) · 4.09 KB
/
IKSwivelHingeJoint.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
using System;
using BEPUutilities;
namespace BEPUik
{
public class IKSwivelHingeJoint : IKJoint
{
/// <summary>
/// Gets or sets the free hinge axis attached to connection A in its local space.
/// </summary>
public Vector3 LocalHingeAxis;
/// <summary>
/// Gets or sets the free twist axis attached to connection B in its local space.
/// </summary>
public Vector3 LocalTwistAxis;
/// <summary>
/// Gets or sets the free hinge axis attached to connection A in world space.
/// </summary>
public Vector3 WorldHingeAxis
{
get { return Quaternion.Transform(LocalHingeAxis, ConnectionA.Orientation); }
set
{
LocalHingeAxis = Quaternion.Transform(value, Quaternion.Conjugate(ConnectionA.Orientation));
}
}
/// <summary>
/// Gets or sets the free twist axis attached to connection B in world space.
/// </summary>
public Vector3 WorldTwistAxis
{
get { return Quaternion.Transform(LocalTwistAxis, ConnectionB.Orientation); }
set
{
LocalTwistAxis = Quaternion.Transform(value, Quaternion.Conjugate(ConnectionB.Orientation));
}
}
/// <summary>
/// Constructs a new constraint which allows relative angular motion around a hinge axis and a twist axis.
/// </summary>
/// <param name="connectionA">First connection of the pair.</param>
/// <param name="connectionB">Second connection of the pair.</param>
/// <param name="worldHingeAxis">Hinge axis attached to connectionA.
/// The connected bone will be able to rotate around this axis relative to each other.</param>
/// <param name="worldTwistAxis">Twist axis attached to connectionB.
/// The connected bones will be able to rotate around this axis relative to each other.</param>
public IKSwivelHingeJoint(Bone connectionA, Bone connectionB, Vector3 worldHingeAxis, Vector3 worldTwistAxis)
: base(connectionA, connectionB)
{
WorldHingeAxis = worldHingeAxis;
WorldTwistAxis = worldTwistAxis;
}
protected internal override void UpdateJacobiansAndVelocityBias()
{
linearJacobianA = linearJacobianB = new Matrix3x3();
//There are two free axes and one restricted axis.
//The constraint attempts to keep the hinge axis attached to connection A and the twist axis attached to connection B perpendicular to each other.
//The restricted axis is the cross product between the twist and hinge axes.
Vector3 worldTwistAxis, worldHingeAxis;
Quaternion.Transform(ref LocalHingeAxis, ref ConnectionA.Orientation, out worldHingeAxis);
Quaternion.Transform(ref LocalTwistAxis, ref ConnectionB.Orientation, out worldTwistAxis);
Vector3 restrictedAxis;
Vector3.Cross(ref worldHingeAxis, ref worldTwistAxis, out restrictedAxis);
//Attempt to normalize the restricted axis.
float lengthSquared = restrictedAxis.LengthSquared();
if (lengthSquared > Toolbox.Epsilon)
{
Vector3.Divide(ref restrictedAxis, (float)Math.Sqrt(lengthSquared), out restrictedAxis);
}
else
{
restrictedAxis = new Vector3();
}
angularJacobianA = new Matrix3x3
{
M11 = restrictedAxis.X,
M12 = restrictedAxis.Y,
M13 = restrictedAxis.Z,
};
Matrix3x3.Negate(ref angularJacobianA, out angularJacobianB);
float error;
Vector3.Dot(ref worldHingeAxis, ref worldTwistAxis, out error);
error = (float)Math.Acos(MathHelper.Clamp(error, -1, 1)) - MathHelper.PiOver2;
velocityBias = new Vector3(errorCorrectionFactor * error, 0, 0);
}
}
}