Skeleton Kinematics - cmu462/Scotty3D GitHub Wiki
A Skeleton
is what we use to drive our animation. You can think of them like the set of bones we have in our own bodies and joints that connect these bones. For convenience, we have merged the bones and joints into the Joint
class which holds the orientation of the joint relative to its parent in its rotation
, and axis
representing the direction and length of the bone with respect to its parent Joint
. Each Mesh
has an associated Skeleton
class which holds a rooted tree of Joint
s, where each Joint
can have an arbitrary number of children.
All of our joints are ball Joint
s which have a set of 3 rotations around the .
Task 2a - Forward Kinematics
Note: These diagrams are in 2D for visual clarity, but we will work with a 3D kinematic skeleton.
When a joint's parent is rotated, that transformation should be propagated down to all of its children. In the diagram below, , it only affects itself because it has no children.
media/forward_kinematic_diagram.jpg
You need to implement these routines in dynamic_scene/joint.cpp
for forward kinematics.
- Joint::getTransformation()
Returns the transformation up to the base of this joint (end of its parent joint).
Joint
is a child class ofSceneObject
which has its own version ofgetTransformation()
that you can use by callingSceneObject::getTransformation()
. As explained above, a joint's transformation is accumulated as you traverse the hierarchy of joints. You should traverse upwards from this joint's parent all the way up to the root joint and accumulate their transformations. Also, make sure to apply the mesh's transformation at the end, accessible withskeleton->mesh->getTransformation()
. Joint::getBasePosInWorld()
Returns the base position of the joint in world coordinate frame.Joint::getEndPosInWorld()
Returns the end position of the joint in world coordinate frame.
Once you have implemented these basic kinematics, you should be able to define skeletons, set their positions at a collection of keyframes, and watch the skeleton smoothly interpolate the motion (see the user guide for an explanation of the interface). The gif below shows a very hasty demo defining a few joints and interpolating their motion.
Note that the skeleton does not yet influence the geometry of the cube in this scene -- that will come in Task 3!
Task 2b - Inverse Kinematics
Single Target IK
Now that we have a logical way to move joints around, we can implement Inverse Kinematics, which will move the joints around in order to reach a target point. There are a few different ways we can do this, but for this assignment we'll implement an iterative method called gradient descent in order to find the minimum of a function. For a function , we'll have the update scheme:
Where is a small timestep. For this task, we'll be using gradient descent to find the minimum of the cost function:
Where  is the position in world space of the target point.
More specifically, we'll be using a technique called Jacobian Transpose, which relies on the assumption that:
Where:
-  is the function media/task4_dev_eq_images/0058.png, where media/task4_dev_eq_images/0059.png is the angle of joint [[media/task4_dev_eq_images/0060.png) around the axis of rotation
is a constant
-  is the Jacobian of [[media/task4_dev_eq_images/0063.png)
Note that here .
Now we just need a way to calcluate the Jacobian of . For this, we can use the fact that:
Where:
- 
is the axis of rotation
-  to the end point of the target joint
For a more in-depth derivation of Jacobian transpose (and a look into other inverse kinematics algorithms), please check out this presentation. (Pages 45-56 in particular)
Now, all of this will work for updating the angle along a single axis, but we have 3 axes to deal with. Luckily, extending it to 3 dimensions isn't very difficult, we just need to update the angle along each axis independently.
Multi-Target
We'll extend this so we can have multiple targets, which will then use the function to minimize:
which is a simple extension actually. Since each term is independent and added together, we can get the gradient of this new cost function just by summing the gradients of each of the constituent cost functions!
You should implement multi-target IK, which will take a std::map
of Joint
s and target points for that joint. Each joint can only have 1 target point.
In order to implement this, you should update Joint::calculateAngleGradient
and Skeleton::reachForTarget
. Joint::calculateAngleGradient
should calculate the gradient of in the x,y, and z directions, and add them to
Joint::ikAngleGradient
for all relevant joints. Skeleton::reachForTarget
should actually do the gradient descent calculations and update the angles of each joint, saving them with Joint::setAngle
. In this function, you should probably use a very small timestep, but do several iterations (say, 10s to 100s) of gradient descent in order to speed things up. For even faster and better results, you can also implement a variable timestep instead of just using a fixed one. Note also that the root joint should never be updated.
A key thing for this part is to remember what coordinate frame you're in, because if you calculate the gradients in the wrong coordinate frame or use the axis of rotation in the wrong coordinate frame your answers will come out very wrong!
Using your IK!
Once you have IK implemented, you should be able to create a series of joints, and get a particular joint to move to the desired final position you have selected.