Skeletal Animations

The skeleton modules provides a way to have 2d skeletal animations in your cocos applications. The process of may be a bit complicated to setup, but using them afterwards is easy, and there are some tools to simplify the process.

To do skeletal animations in cocos you will have to:

  • Create a skeleton

  • Create one or many skins

  • Create all the animations you want

Then, to use them you will have to:

  • Load the skeleton

  • Load the skin

  • Add a Skin cocos node to your scene

  • Load the animations

  • Use the Anim action on the skin node to play animations.

Creating Skeletons and skins

The skeleton file

Skeletons are defined in .py files. The first thing you will do is create the root bone for your skeleton. Here we create what is going to be the body of our human skeleton as follows:

doc/programming_guide/skeleton/root_bone.py:

from cocos.skeleton import Bone, Skeleton

root_bone = Bone('body', 70, -180.0, (0.00, 0.00))

skeleton = Skeleton(root_bone)

This reads like: Create a bone labeled ‘body’ that is 70 pixels long, is rotated by -180 degrees and starts on (0,0). We rotate the bone because we want to be able to move its shoulders and not its legs, so the fixed part of the skeleton will be its waist, ie, the origin of the root bone.

You can see the skeleton you have created using with a small piece of code.

from doc/programming_guide/skeleton/show_root_bone.py:

from cocos import skeleton

# import the skeleton we have created
import root_bone

class TestLayer(cocos.layer.Layer):
    def __init__(self):
        super( TestLayer, self ).__init__()

        x,y = director.get_window_size()

        # create a ColorSkin for our skeleton
        self.skin = skeleton.ColorSkin(root_bone.skeleton, (255,0,0,255))

        # add the skin to the scene
        self.add( self.skin )
        x, y = director.get_window_size()
        self.skin.position = x/2, y/2

It will look like this:

../../_images/show_root_bone.png

The important new concept here is the skin. Skins are responsible for drawing the skeletons. Here we use the basic skin, ColorSkin, that just draws a colored line over the bone. A skin, like everything else, is just a cocosnode, so you can add it to your node to place it on screen.

The skin file

Now, we want to skin this skeleton with some images, so we create a skin file. The skin file will look like this:

doc/programming_guide/skeleton/root_skin.py:

skin = [
    ('body', (25, 91), 'gil-cuerpo.png', True, True, 0.5),
  ]

This reads like: This skin ha an image for the bone labeled ‘body’, this image has an offset of (25, 91) from the bone origin, the image file is ‘gil-cuerpo.png’ (provided by cocos), its flipped on its x axis, its flipped on its y axis (remember we rotated the bone by -180), and its scaled to half its size.

Now we can show the skeleton with the skin we have just created.

from doc/programming_guide/skeleton/show_root_skin.py:

# import the skeleton we have created
import root_bone
import root_skin

class TestLayer(cocos.layer.Layer):
    def __init__(self):
        super( TestLayer, self ).__init__()

        x,y = director.get_window_size()

        # create a ColorSkin for our skeleton
        self.skin = skeleton.BitmapSkin(root_bone.skeleton, root_skin.skin)

        # add the skin to the scene
        self.add( self.skin )
        x, y = director.get_window_size()
        self.skin.position = x/2, y/2

It will look like this:

../../_images/show_root_skin.png

The skeleton editor

At this point, we can use the first tool provided, the skeleton editor. You start the editor like this:

tools$ python skeleton/skeleton_editor.py ../doc/programming_guide/skeleton/root_bone.py ../doc/programming_guide/skeleton/root_skin.py

It will look like this:

../../_images/root_editor.png

There you can see the skin part of your skeleton and several control points. Control points can be dragged. If a control point is on top of another control point, you can change your selection using the scroll weel. The current control point has a small halo.

In this example you can see the red control point, which controls the rotation of the bone, the two blue control points, which control the position of the bone and skin part, and the green control point (behind one of the blue ones) which controls the skeleton position.

Pressing ‘s’ will save the changes you have made to the skeleton and skin file. This will overwrite the .py files, so b carefull to just pur the data of the skeleton and skin in this files.

This is a helpfull tool in creating your skins, as the only way to know the real offsets between your skeleton parts and the image the artist has creates is visually.

Now we add more bones and skin parts to the skeleton. Bones, other than the root bone, are attached to other bones and thei position and rotation are relative to its parent bone position and rotation.

doc/programming_guide/skeleton/body_and_arm.py:

from cocos.skeleton import Bone, Skeleton

def Point2(*args): return args

root_bone = Bone('body', 70, -180.0, Point2(0, 0)).add(
    Bone('upper_arm', 30, 120, (0, -70)).add(
        Bone('lower_arm', 30, 30, (0, -30))
    )
)

skeleton = Skeleton(root_bone)

Here we abuse the fact that the function ‘add’ returns the parent bone, so you can chain call to add and have and indentation that resembles the hierarchical structure of the skeleton.

The upper arm is 30 pixels long and is placed at the end of the body by offsetting if -70 pixels. The lower arm is also 30 pixels long and is also placed at the end of its parent, the upper arm. Bone length doesnt affect the transofrmations or placement of other bones, but it useful to see what you are creating.

The skeleton we have just created looks like this:

../../_images/body_and_arm.png

Now we add the skin. We wont care about the positions or flipping, as we will be changing that parameters when looking at it.

doc/programming_guide/skeleton/body_and_arm_skin.py:

skin = [
    ('body', (25, 91), 'gil-cuerpo.png', True, True, 0.5),
    ('lower_arm', (0, 0), 'gil-mano2.png', False, False, 0.5),
    ('upper_arm', (0, 0), 'gil-brazo1.png', False, False, 0.5),
  ]

The mess we have just created will look like this:

../../_images/body_and_arm_skin_pre.png

We now have to use the editor to drag the parts until it looks like this:

../../_images/body_and_arm_skin_final.png

NOTE: When moving the control points, things look weird. Its because you are actually only aplying a translation to the point. But the translation is then aplied over the transformed point. It looks crazy, but you can get used to it if you play with it for a while. And if you dont, you can just submit a patch.

It is important to move the bones, not just the skin, so the shoulder and elbow move at the right locations. You can test that rotating the bones and making sure that the images look good in all positions.

You then press the ‘s’ key to save.

You should continue this way until you have the full skeleton and skin defined.

You can see and use the sample skeleton and skin provided in the test directory:

tools$ python view_skeleton.py ../test/sample_skeleton.py

tools/skeleton$ python skeleton_editor.py ../../test/sample_skeleton.py ../../test/sample_skin.py

Animations

Now that you have your skeletons, its time to animate them. An animation has a list of keyframes, ie, skeleton poses and times. When you play an animation, cocos will pose your skeleton according to the current time, interpolating between the previous and next frame.

The animator editor

Start the animation editor doing:

tools/skeleton$ python animator.py ../../test/sample_skeleton.py --skin ../../test/sample_skin.py ../../test/SAMPLE.anim
../../_images/animator.png

On the top you can see the timeline with the keyframes in blue and the current position marked in yellow.

You can also see the first keyframe, with its skin and control points.

You can control the animator with the following keys:

  • S: Save the current animator

  • LEFT: move $TICK_DELTA time to the left

  • RIGHT: move $TICK_DELTA time to the right

  • PLUS: add a keyframe in the current position

  • MINUS: remove the keyframe in the current position

  • PAGE UP: Move to the previous keyframe

  • PAGE DOWN: Move to the next keyframe

  • INSERT: insert $TICK_DELTA time at the current position

  • DELETE: delete $TICK_DELTA time at the current position

  • HOME: Go to the start of the animation

  • END: Go to the end of the animation

  • SPACE: Play the animation

When standing on a keyframe you will see some control points, that you can move to edit the keyframe.

$TICK_DELTA is 1/16 of a second.

Playing animations

You can play an animation on a skin doing:

from test/test_skeleton_anim.py:

class TestLayer(cocos.layer.Layer):
    def __init__(self):
        super( TestLayer, self ).__init__()

        x,y = director.get_window_size()
        self.skin = skeleton.BitmapSkin(sample_skeleton.skeleton, sample_skin.skin)
        self.add( self.skin )
        x, y = director.get_window_size()
        self.skin.position = x/2, y/2
        anim = cPickle.load(open("SAMPLE.anim"))
        self.skin.do( cocos.actions.Repeat( skeleton.Animate(anim) ) )

The Animate action has the following parameters: recenter, recenter_x, recenter_y. This control how the skeleton position is moved. If you run two walk left animations without translation you will see a player move left, go to the origin and move left again. if you use translation, the player will just move left twicenas far.

Also, animations can be flipped, which makes the character look to the other side.