-
Notifications
You must be signed in to change notification settings - Fork 30
Architecture Guide
This is a document describing the ideal state of the Mo' Bends architecture. This is how it should all work in the end, and is the current target for the 1.0.0 release of Mo' Bends for 1.16.4.
In the new architecture, I chose to separate the assets (animations, models, state-machines) from the systems. This is done because assets shouldn't have to change between versions of Minecraft, they shouldn't depend on which version of Minecraft the animations are being played and applied on. The systems on the other hand work hand-in-hand with the Minecraft code in order to bring these assets to life. The code should be general, minimal and testable so that porting is easier.
These asset files can be provided by both resource-packs as well as mods, which allows for the following:
- Mods can provide their own metadata or even animations that will show up when used alongside Mo' Bends,
- Resource-pack creators can create animation-sets for whatever mob they want, even outside of the the default animation-set (as long as they provide mutation data),
A resource-pack that wants to add animations for mobs can contain the following:
- Animator files - this is a file in a custom binary format holding animation data and a description of a finite-state-machine that controls when certain animations will get played. They have to specify which mutator they're basing their animators on to avoid conflicts and increase compatibility. This coupling could in theory allow users to swap between mutators and their associated animation sets on the fly.
- Mutator files - .json files containing instructions on how a model should be mutated (what body parts should change and how) in order to then have a wider range of motions the character is able to perform. Default mutators for some mobs are shipped with Mo' Bends, so you can just base your animations on those.
- Metadata files - .json files which the system is going to use to translate body part names into internal code field names. This separation is useful since the names of body parts in-code change frequently from version to version, and we want to keep these resource-packs as cross-version as we can. These files are rarely going to have to get created by resource-pack creators, unless they want to animate a mob outside of vanilla.
In the following resource-pack file structure, the <modid>
parameter should be replaced by:
- If creating a mod, that mod's id
- If creating a resource-pack:
mobends
└── assets
└── <modid>
├── bendsanimations
│ ├── wolf_idle.bendsanim
│ ├── wolf_sitting.bendsanim
| ├── player_walk.bendsanim
| └── (...)
|
├── bendsanimators
│ ├── wolf.bends
| ├── player.bends
| └── (...)
|
└── bendsmutators
| ├── wolf.json
| ├── player.json
| └── (...)
|
└── bendsmeta
├── wolf.json
├── player.json
└── (...)
{
/**
* A unique identifier of this mutator. No two mutators can exist with the same id. To avoid that conflict,
* please adhere to the following standard: `[author/name of the pack]/[location of mob]`
*/
"id": "examplepack/minecraft:wolf",
"target": "minecraft:wolf", // Which mob we're mutating. The ID above should match this location.
/**
* Describe how you'd like to change the existing parts of a model.
*/
"mutate": {
"bodyPart1": {
"parent": "root", // [Optional]
/**
* [Optional]
* Typically used when this part is part of a bigger whole, and the 'extentionPart1'
* is the continuation of that bigger whole. Used for example by arms and item placement,
* where we expect the end of the "arm" to be at the hand, instead of mid-way through.
*/
"extendedBy": "extentionPart1",
"position": [0, 1.5, 2], // The pivot position of this body part
"boxes": [
{
"position": [-3, -3.5, -8], // The offset position of this box
"dimensions": [6, 6, 9], // The dimensions (size)
/**
* [Optional] Each inner field (recursive) is also optional. Used for offsetting/rotating faces to alter
* their UV coordinates relative to the standard cube mapping.
*/
"faces": {
"TOP": { "offsetX": 9, "offsetY": 6 },
"BOTTOM": { "offsetX": -8, "offsetY": 6, "rotate": "HALF_TURN" },
"LEFT": { "offsetX": -3, "offsetY": -3, "rotate": "CLOCKWISE" },
"RIGHT": { "offsetX": 0, "offsetY": -3, "rotate": "COUNTER_CLOCKWISE" },
"FRONT": { "offsetX": 1 },
"BACK": { "offsetY": -9 }
}
}
// ...
]
}
// ...
},
/**
* Describe what new parts you'd like to add to the model.
*/
"add": {
"root": {
"position": [0, 0, 0]
},
"extentionPart1": {
"parent": "bodyPart1", // [Optional]
"position": [0, 2, -4], // The pivot position of this body part
"textureOffset": [0, 12], // The top-left origin of where cube mapping should start.
"boxes": [
{
"position": [-1.5, 0, -4], // The offset position of this box
"dimensions": [3, 1, 4], // The dimensions (size)
/**
* [Optional] (Same as the previous body part)
*/
"faces": { /* Same as the previous body part */}
}
// ...
]
}
}
}
Be very careful when providing those, as it may cause conflict when using multiple resource-packs. Only create when absolutely necessary. If you find the need to create one, see if there is any other resource pack mutating the same mob as you plan to, and agree on using the same mapping as in their metadata files.
{
"target": "minecraft:wolf", // Which mob we're providing metadata for.
"parts": {
"body": { "fieldName": "bodyField" },
"head": { "fieldName": "headField" },
"mane": { "fieldName": "upperBodyField" },
"leg1": { "fieldName": "leg0Field" },
"leg2": { "fieldName": "leg1Field" },
"leg3": { "fieldName": "leg2Field" },
"leg4": { "fieldName": "leg3Field" },
"tail": { "fieldName": "tailField" }
// ...
}
}
Finite state machines allow us to describe how characters should react to their current state and to the state of their world. The reactions have to be purely visual, as in animations (so no additional interaction and/or functionality from mobs). These finite state machines are described in binary form, contained within animator files.
Only one mutator per model can be applied at a time, and we assume there's mostly a one-to-one correspondance between an entity type and a model instance. We can therefore limit the number of applied mutators with the same target to at most one.
- Picking an atribtrary mutator, unless a mutator id has been specified in the config. (solution for the first implementation)
- ... (?)
We need a way to change which mutators are applied through the UI.
- A list of loaded mutators categories by their target, each labeled by the resource packs they come from (maybe the filename). The user can then click on a "Set default" button to use that particular mutator for that target by default.
The whole idea of BendsPacks is to extend the default animation set of Mo' Bends. That means that resource pack embedded animators shouldn't have to reinvent the wheel and create each and every animation. They could for example create just one, which plays at a specific condition but falls back to the default animator otherwise. This could be achieved by creating a "Fall-Through" node, which does nothing but letting the previous layer take the spotlight. That means that if an animation set doesn't have any "Fall-Through" nodes and is non-empty, it totally overrides the previously applied animators (which means none of the previous animators have to be evaluated). Let's call this type of animator "overriding".
How do we pick which overriding animator should be the one to get evaluated? And what about the other animators, in what order should they get applied?
- For each mutator, choose an arbitrary overriding animator as the primary one, and keep the non-overriding animators in an arbitrary order. This shouldn't be a problem for the first versions of the system as we don't expect many overriding animators to be made. (solution for the first implementation)
- ... (?)
A user could want to disable certain animations if they don't like their look or if it breaks some other functionality. Right now we only allow disabling animations on the level of whole targets.
Features
Advanced usage
Internal systems (1.X)
Internal systems (2.X)