MonoSkelly is a library + editor + file format for skeleton-based animations in MonoGame. It implements a complete animation system as a new set of classes, decoupled and independent of MonoGame's Models and Content Manager.
To use MonoSkelly:
- Open the Skelly Editor (included in this repository).
- Build a skeleton from an hierarchy of bones (a bone = offset + scale + rotation from parent bone).
- Define animations by setting key poses to interpolate between.
- Save the skeleton file, which you will later use in your game with the MonoSkelly.Core library.
Using MonoSkelly is quite simple too. Start animation sequence, update with delta-time every frame, and query bone transformations to use with your 3d models.
In addition to the editor, this repository contains a demo project you can check out. It renders a player character with simple model transformations, ie without weights, similar to how early 3d games where made.
To use MonoSkelly, fetch the NuGet Package with the NuGet manager, or by running the following command in Visual Studio:
Install-Package MonoSkelly
You can also clone the repo and build the projects yourself.
This repository contains three projects:
The library itself, which is the dll you get from the NuGet and everything you need to use MonoSkelly in your game or applications.
A demo project that illustrate how to use MonoSkelly.
It animates a simple fighter model with sword and shield that can walk around and attack. It doesn't use weights and all calculations are done on CPU.
An editor program (exe) to create and animate skeletons. It comes with an example human skeleton you can load and play with.
The editor is open source, so you can incorporate it in your own projects or modify it to better suit your needs.
In this section we'll learn how to use MonoSkelly via the package API.
If you want to learn more about the editing API and not just how to play animations (for example if you want to generate animations at runtime), check out the editor source code or read the details in the API section.
Let's take a look at a basic example on how to use MonoSkelly.
Please create an empty MonoGame project and add the MonoSkelly NuGet package to it. Then, copy the content folder from MonoSkelly.Demo/Content/tutorial
and add it to your own project content. This will provide you with a basic skeleton animation and models we'll use in this example.
Be sure to set 'robot.ini' build action to just 'Copy', as this file is not meant to be built.
At the end of the example, you should have a little dancing robot that looks something like this:
Yes, it's incredible. I know.
Now to the code! Add the following members to your Game class:
// camera for view & projection metrices
MonoSkelly.Core.Camera _camera;
// player models (body parts, key = bone alias)
Dictionary<string, Model> _playerParts;
// player skeleton and active animation
MonoSkelly.Core.Skeleton _playerSkeleton;
MonoSkelly.Core.AnimationState _animation;
And let's start by loading the models we'll use to render the robot with. Note that the keys in this dictionary match the aliases defined in the skeleton file, so model "head" will be attached to bone "head", model "body" to bone "body", etc.
protected override void LoadContent()
{
// load player models
_playerParts = new Dictionary<string, Model>();
_playerParts["head"] = Content.Load<Model>("tutorial/head");
_playerParts["body"] = Content.Load<Model>("tutorial/body");
_playerParts["arm_left"] = Content.Load<Model>("tutorial/arm");
_playerParts["arm_right"] = Content.Load<Model>("tutorial/arm");
_playerParts["leg_left"] = Content.Load<Model>("tutorial/leg");
_playerParts["leg_right"] = Content.Load<Model>("tutorial/leg");
// fix models base transform (messed up in export)
foreach (var model in _playerParts)
{
model.Value.Root.Transform = Matrix.CreateRotationY(MathHelper.Pi / 2);
}
}
We'll also create a camera, which is a utility class provided by MonoSkelly to help us build the View and Projection matrices:
_camera = new MonoSkelly.Core.Camera(_graphics);
And now to load our skeleton animations, we'll add the following code:
_playerSkeleton = new MonoSkelly.Core.Skeleton();
_playerSkeleton.LoadFrom("Content/tutorial/robot.ini");
And finally, we'll create an animation instance to play the 'idle' animation:
MonoSkelly.Core.AnimationState _animation = _playerSkeleton.BeginAnimation("idle");
That's it, the stage is now set for rendering. Let's write our Draw() method:
protected override void Draw(GameTime gameTime)
{
GraphicsDevice.Clear(Color.CornflowerBlue);
// update camera
_camera.LookAt = new Vector3(0, 15, 0);
_camera.Position = new Vector3(0, 45, -45);
_camera.Update();
// advance animation by 'deltaTime' (usually can be gameTime.ElapsedGameTime.TotalSeconds)
_animation.Update((float)gameTime.ElapsedGameTime.TotalSeconds);
// player world model (you can translate this to move player around)
var playerWorldMatrix = Matrix.Identity;
// now draw the player models
foreach (var part in _playerParts)
{
var bone = _animation.GetBoneTransform(part.Key);
part.Value.Draw(bone * playerWorldMatrix, _camera.View, _camera.Projection);
}
base.Draw(gameTime);
}
And if the skeleton and models are properly set, you can see the robot playing the idle animation!
There are two key methods to note in the code above:
animation.Update
is the method we use to advance the animation by a delta time every frame. It also has an overload method with additional out parameters, which provides extra information like how many steps were passed, and whether or not we finished the animation cycle in this frame.
animation.GetBoneTransform
is what we use to query the bones transformations, and we use these matrices with our models to animate them.
GetBoneTransform()
can either accept the bone's full path (for example, root/torso/arms/upper_arm
), or the bone alias, if defined.
In case your code didn't work, you can compare it to the following full example code, also found in MonoSkelly.Demo/UsageExampleCode.cs
:
internal class UsageExampleCode : Game
{
private GraphicsDeviceManager _graphics;
// camera for view & projection metrices
MonoSkelly.Core.Camera _camera;
// player models (body parts, key = bone alias)
Dictionary<string, Model> _playerParts;
// player skeleton and active animation
MonoSkelly.Core.Skeleton _playerSkeleton;
MonoSkelly.Core.AnimationState _animation;
public UsageExampleCode()
{
_graphics = new GraphicsDeviceManager(this);
Content.RootDirectory = "Content";
IsMouseVisible = true;
}
protected override void Initialize()
{
_camera = new Core.Camera(_graphics);
base.Initialize();
}
protected override void LoadContent()
{
// load player models
_playerParts = new Dictionary<string, Model>();
_playerParts["head"] = Content.Load<Model>("tutorial/head");
_playerParts["body"] = Content.Load<Model>("tutorial/body");
_playerParts["arm_left"] = Content.Load<Model>("tutorial/arm");
_playerParts["arm_right"] = Content.Load<Model>("tutorial/arm");
_playerParts["leg_left"] = Content.Load<Model>("tutorial/leg");
_playerParts["leg_right"] = Content.Load<Model>("tutorial/leg");
// fix models base transform (messed up in export)
foreach (var model in _playerParts)
{
model.Value.Root.Transform = Matrix.CreateRotationY(MathHelper.Pi / 2);
}
// load skeleton and init animation
_playerSkeleton = new MonoSkelly.Core.Skeleton();
_playerSkeleton.LoadFrom("Content/tutorial/robot.ini");
_animation = _playerSkeleton.BeginAnimation("idle");
}
protected override void Update(GameTime gameTime)
{
if (GamePad.GetState(PlayerIndex.One).Buttons.Back == ButtonState.Pressed || Keyboard.GetState().IsKeyDown(Keys.Escape))
Exit();
base.Update(gameTime);
}
protected override void Draw(GameTime gameTime)
{
GraphicsDevice.Clear(Color.CornflowerBlue);
// update camera
_camera.LookAt = new Vector3(0, 15, 0);
_camera.Position = new Vector3(0, 45, -45);
_camera.Update();
// advance animation by 'deltaTime' (usually can be gameTime.ElapsedGameTime.TotalSeconds)
_animation.Update((float)gameTime.ElapsedGameTime.TotalSeconds);
// player world model (you can translate this to move player around)
var playerWorldMatrix = Matrix.Identity;
// now draw the player models
foreach (var part in _playerParts)
{
var bone = _animation.GetBoneTransform(part.Key);
part.Value.Draw(bone * playerWorldMatrix, _camera.View, _camera.Projection);
}
base.Draw(gameTime);
}
}
Say your player now attacks, and you want to render the attacking animation.
To do so, we just need to change our animation state to the 'attack' animation (note: the robot skeleton does not come with an attack animation, so you need to build one yourself first):
_animation = _playerSkeleton.BeginAnimation("attack");
And when done, we can switch back to idle:
_animation = _playerSkeleton.BeginAnimation("idle");
If you tried switching between animations, you probably noticed that something's not right.
While both animations play smoothly, the transition between them is not. Every time a new animation starts, the bones 'jump' to the start pose of the new animation.
That's because in order to get a smooth transition between animations, you need to blend them. Animation blending is not unique to MonoSkelly, it's a common practice in 3d graphics, in which we lerp the transformations from one animation to the next, usually while playing both.
To blend animations, MonoSkelly offers a helper class called AnimationsBlender. Lets take a look on how we use it:
var idleAnimation = _playerSkeleton.BeginAnimation("idle");
var attackAnimation = _playerSkeleton.BeginAnimation("attack");
_animationsBlender = new MonoSkelly.Core.AnimationsBlender(idleAnimation, attackAnimation);
And now when you draw your models, instead of using the animation state, use the blender:
// advance animation by 'deltaTime' (usually can be gameTime.ElapsedGameTime.TotalSeconds)
_animationsBlender.Update(deltaTime);
// advance the transition from 'idle' to 'attack' (0 = idle animation, 1 = attack)
_animationsBlender.BlendFactor += deltaTime;
// draw the player models
foreach (var part in _playerParts)
{
var bone = _animationsBlender.GetBoneTransform(part.Key); // <-- will return blended transformations between the animations
part.Value.Draw(bone * playerWorldMatrix, _camera.View, _camera.Projection);
}
Normally, bones are identified by their full path, meaning that if you have the following bones hierarchy: root -> body -> left -> arm, the arm bone will be identified as root/body/left/arm
. That way you can have both root/body/left/arm
and root/body/right/arm
without having a collision on the arm
bone name.
Full paths are even more useful as your hierarchy grows deeper, for example if you have root/body/left/arm/palm/fingers/...
and you want to have 'arm', 'palm' and same set of fingers on both sides.
However, using full paths is not always comfortable, as it creates an unwanted coupling between the skeleton structure and your code. What if you want to use a slightly modified skeleton, that has a torso
bone between body
and left
? Now your path is root/body/torso/left/arm
, and you need to change your code accordingly.
Aliases solve exactly that, by decoupling the code from the skeleton.
In essense, an alias is a short unique name you can attach to a bone and later use in code, without knowing the bone actual path.
In the example above we would attach the alias left_arm
to root/body/left/arm
and right_arm
to root/body/right/arm
. That way we can render both arms without knowing the structure, the only requirement is that the skeleton will have have a 'left_arm' and 'right_arm' bones.
When possible, always prefer to use aliases over full paths. It will also make it easier for players to mod your game, if you desire that.
For debug purposes, you can draw the skeleton itself. First, you need to set a model to use when drawing bones:
MonoSkelly.Core.Skeleton.SetDebugModels(null, Content.Load<Model>("bone"), null);
Now we can draw the skeleton (we'll draw 'idle' animation with elapsed time of '_animationProgress'):
_playerSkeleton.DebugDrawBones(_camera.View, _camera.Projection, playerWorldMatrix, "idle", _animationProgress, true, null);
Note that drawing bones don't happen automatically, you need to set which bones to draw and how to draw them in the editor. More on that later.
A warning before we begin! - at the time of writing these lines the editor is still very new and not tested enough. Please for the love of god, save your file as often as possible.
Lets explore the editor for a little bit, to learn how to make animations.
Try to build and run the Editor project. When opened, you should see the following layout (note: in this example we also loaded the sample skeleton 'human.ini'):
Start a new project, and lets build something!
First step is to define your skeleton in it's default pose.
Use the 'New Bone' button (from the left panel) to create a new bone and build your skeleton structure. Use the bone transformations to move, scale and rotate the bones (usually you only need rotation and offset, scale is only if you want to scale models while animating them).
Its a common practice to define the default pose as the famous "T-Pose", but its not mandatory with MonoSkelly.
Note that bones will not be visible in editor like in the layout screenshot. Instead you'll only see a small sphere representing the bone's origin point (aka 'handle').
To add a bone display model, check the 'Render Bone In Editor' and define its scale and offset to match the models you actually want to use in-game.
Let's create an animation with the new animation button on the animations panel (usually on the right):
After creating an animation select it, and add a step by clicking on the new step button. Give it any name, or leave empty. Step names are not mandatory, but they are useful for clarity, and you can use them to trigger stuff while playing the animation in your game (for example you can set a step called 'attack_strike' that represent the actual hit, and only deliver damage when reach it).
Once you have a valid animation selected and there's at least one step in it, the rest of the animation panel options will unlock and become interactable:
Note the little play button under the timeline. You can click it to play the animation, but since you only have one step and didn't change any bone yet, clicking it now won't do much.
You can now start working on your animation by moving the bones to the pose you want while the step is selected. Then add another step and create a new pose, until you define all the key poses of your animation. When playing it, MonoSkelly will interpolate transformations between the poses you defined.
Note that every step has a 'duration' property. If a step duration is 1.0 seconds, for example, it means that MonoSkelly will take exactly one second to interpolate between the begining of this step and the bones transformations of the next step.
If your timeline points on an interpolated frame (ie between steps), the transformations panel will be locked, as you can't edit bones between steps. You can use the steps selection dropdown to quickly go back to one of the valid steps.
When you finish your animations save your skeleton via the 'File' menu at the top-left corner. Output file will appear in the saves folder, and you can copy it to your project folder and load it with the LoadFrom() API.
That's it! You should now know how to use the Editor to create animations, and then use and blend them in your game. This would be a good time to check the demo project in depth, or to proceed to the API section in this doc.
This section describe the main classes and their API. This does not cover everything, only the main classes and members you'll be using most of the time. For the full API, its best to check the code itself.
If you jumped here before reading the How To Use section, its best to go back to the basic example first.
A helper class to define bone's transformations (position, scale, rotation) and convert it to a matrix.
This is the main class we use in MonoSkelly. Note that a Skeleton does not represent an instance, but a type. Ie you won't have a skeleton instance per enemy instance, but you'll probably have a Skeleton per enemy model type, which will be used to animate all the enemies that use that model.
Lets explore the Skeleton's main methods:
Set the models to use for bones and handles display while in editor / debug draw.
Return an AnimationState instance to start playing a new animation. This is what you use to animate the skeleton.
Set a bone alias.
Check if this Skeleton contains an alias.
You can use this to decide if to render a helmet, for example, based on whether or not the given skeleton contains an alias called "head_gear".
Set the default pose bones transformation.
Save Skeleton to file.
Load Skeleton from file.
Define a new bone.
Draw the skeleton bones, using the models you set with SetDebugModels
.
Draw the skeleton bone handles, using the models you set with SetDebugModels
.
A quick and dirty way to get a bone transformation. Best not to use this in production.
A running animation instance. Use this to animate models and query bones.
Reset animation back to start.
Get bone transformations.
Updates the animation. You must call this with delta time to progress the animation.
Set if this animation will repeat itself in a cycle.
Get animation name.
A utility class to blend two animations together.
Animation to blend from.
Animation to blend to.
Blending factor (0.0 - 1.0) between the animations.
- 0.0 = 100% BlendFrom weight, 0% BlendTo weight.
- 1.0 = 0% BlendFrom weight, 100% BlendTo weight.
Optional callback that will be called whenever an animation finishes a cycle.
Switch 'from' and 'to' animations.
Get bone transformations, blended between the animations based on factor.
Update both animations by delta time, without changing blending factor. You need to update the blending factor separately.
A utility class to help us build View and Projection matrices.
Rebuild the View and Projection matrices.
If set, will use this as screen size instead of querying the graphic device.
Select camera type (Perspective / Orthographic).
Set FOV.
Set near and far clip planes.
Optional lookat vector.
Camera position.
Camera up vector.
Camera rotation on X and Y axis. Only works if LookAt is null.
Get the Projection matrix.
Get the View matrix.
Get the View * Projection matrix.
Get camera view frustum.
Create a ray from mouse position.
Create a ray from 3d point.
Create a ray from 2d point.
MonoSkelly files are saved as a plaintext ini files. It may not be the most efficient format, but it has the advantages of being readable, easy to edit with just notepad, and you can actually compare changes with simple text comparison tools.
Looking at the sample human.ini
file should tell you everything you need to know about the format, but lets review some of the sections a MonoSkelly contains here:
[bones]
This section contains all the bones in their default pose. Every bone have its path (which also contain its parent), and the bone's transformations (position, scale and rotation).
[meshes]
This section contains the meshes we use to display the bones in the editor. Its for debug purposes and editor only.
[aliases]
This section contains all the bone aliases. Keys are the alias, values are bone paths.
[animations]
This section contains just the names of the animations this skeleton support.
For every animation in the list, we'll have a corresponding [animation_xxx]
section to describe the animation itself (xxx being animation name).
[animation_xxx]
Describe the specific animation sequence.
For every animation we have few metadata fields, like steps count and if the animation is repeating, and a set of animation steps.
Every animation step contains a list of bones and the transformations to apply on every bone in this step.
To slightly reduce file size, instead of using bone paths here we use indices, which should match the order in which the bones appear under the [bones]
section.
- Fixed editor crashing when adding bone to default pose and switching to animation.
- Fixed animation clone to also copy
Repeats
and steps steps interpolation modes. - Fixed editor to lock transformations while animation with no steps is selected.
- Made the animations dropdown sorted by name.
- Added 'SetNextAnimation' helper method to animations blender.
MonoSkelly is distributed under the MIT license, so use it for whatever.