Skip to content

SpriteMesh is a plugin for Godot that allows you to create 3D meshes based on a 2D sprite.

License

Notifications You must be signed in to change notification settings

98teg/SpriteMesh

Repository files navigation

Sprite Mesh

Sprite Mesh Banner

Warning The main branch of this repository now supports Godot 4.x. If you are looking for the Godot 3.x version, please check this branch.

Introduction

SpriteMesh is a plugin for Godot that allows you to create 3D meshes based on a 2D sprite. It adds two new classes, SpriteMesh and SpriteMeshInstance. SpriteMesh is a Resource that contains an array of meshes and their material. SpriteMeshInstance, which inherit from MeshInstance, is used to create the meshes based on the sprite.

SpriteMeshInstance can also be used to display the generated meshes on the scene. It has the benefit over MeshInstance in that it adds support for animations in which each frame is a different mesh, as seen in the example below.

Animation RPG Demo

The algorithm SpriteMeshInstance uses to generate the meshes is a bit demanding. It is for this reason that I do not recommend executing it frequently. However, it has the benefit that the mesh it produces tends to optimize the number of tris. In this image, you can see the result of this algorithm.

Algorithm Result

How to use it

There are two ways of using this plugin, via the editor or code. For both of them, you would need to include the folder sprite_mesh of this project into the addons directory of your Godot project. Then click the Project dropdown, select Project Settings... and go to the Plugins tab. Lastly, click on the Active check box at the SpriteMesh row.

Using the editor

I recommend this method if you don't need to generate the meshes procedurally. Using the editor, the algorithm would not be executed at runtime.

  1. Instantiate a SpriteMeshInstance in your scene.
  2. Assign the texture for your model.
  3. Change any of the properties, if needed. When you change a property, you need to wait three seconds for the editor to update the meshes. This behavior is intended, as it provides a better user experience.
  4. Save or copy the generated mesh if you want to use it in MeshInstance nodes.
  5. Save or copy the generated material if you want to use it in MeshInstance nodes. Remember to make it unique if you reuse this SpriteMeshInstance to generate other meshes.

If you want to use animations and pretend to use SpriteMeshInstance nodes in your scene instead, you can save or copy the SpriteMesh and assign it to other SpriteMeshInstance nodes. It is more memory efficient than creating another SpriteMeshInstance and setting the same texture.

Editor Usage

Using code

I recommend this method if you need to generate the meshes procedurally. Bellow is an example of how to create them.

Code Usage

Even if this option is available, I recommend only executing it on methods that are not called frequently, such as _ready. If you want to, for example, flip a character sprite, it is better to just rotate the model than changing the flip_h property and regenerating it. The only properties meant to change frequently at runtime are frame and frame_coords. And as such, they don't require to call update_sprite_mesh to be applied.

Class SpriteMesh

Description

SpriteMesh is a Resource that contains an array of meshes and their material.

Properties

Type Name Default value
StandardMaterial3D material
Array[ArrayMesh] meshes []

Properties Descriptions

StandardMaterial3D material

The meshes' material.

void set_material(StandardMaterial3D material)

StandardMaterial3D get_material()

Array[ArrayMesh] meshes

Array of meshes. Each mesh of the array represents a frame of the animation.

void set_meshes(Array[ArrayMesh] meshes)

Array[ArrayMesh] get_meshes()

Class SpriteMeshInstance

Description

SpriteMeshInstance, which inherit from MeshInstance, is used to create the meshes based on the sprite. It is inspired by Sprite3D, so many of its properties behave similarly.

Properties

Type Name Default value
float alpha_threshold 0.0
Vector3.Axis axis 2
bool centered true
float depth 1.0
bool double_sided true
bool flip_h false
bool flip_v false
int frame 0
Vector2i frame_coords Vector2i(0, 0)
SpriteMesh generated_sprite_mesh
int hframes 1
Vector3 offset Vector2(0, 0)
float pixel_size 0.01
bool region_enabled false
Rect2 region_rect Rect2(0, 0, 0, 0)
Texture2D texture
float uv_correction 0.0
int vframes 1

Methods

Returned type Declaration
ArrayMesh get_mesh_with_index(int index)
void update_sprite_mesh()

Properties Descriptions

float alpha_threshold = 0.0

The maximum value of alpha for the algorithm to not render a given pixel.

void set_alpha_threshold(float value)

float get_alpha_threshold()

Vector3.Axis axis = 2

The direction in which the front of the mesh faces.

void set_axis(Vector3.Axis value)

Vector3.Axis get_axis()

bool centered = true

If true, mesh will be centered.

void set_centered(bool value)

bool is_centered()

float depth = 1.0

Depth of the mesh, measured in pixels.

void set_depth(float value)

float get_depth()

bool double_sided = true

If true, mesh can be seen from the back as well, if false, it is invisible when looking at it from behind.

void set_draw_flag(DrawFlags flag, bool enabled)

bool get_draw_flag(DrawFlags flag)

bool flip_h = false

If true, mesh is flipped horizontally.

void set_flip_h(bool value)

bool is_flipped_h()

bool flip_v = false

If true, mesh is flipped vertically.

void set_flip_v(bool value)

bool is_flipped_v()

int frame = 0

Current frame to display from sprite sheet. hframes or vframes must be greater than 1.

void set_frame(int value)

int get_frame()

Vector2i frame_coords = Vector2i(0, 0)

Coordinates of the frame to display from sprite sheet. This is as an alias for the frame property. hframes or vframes must be greater than 1.

void set_frame_coords(Vector2i value)

Vector2i get_frame_coords()

SpriteMesh generated_sprite_mesh

The result of the algorithm. It would generate automatically in the editor, or after calling update_sprite_mesh in code.

void set_generated_sprite_mesh(SpriteMesh value)

SpriteMesh get_generated_sprite_mesh()

int hframes = 1

The number of columns in the sprite sheet.

void set_hframes(int value)

int get_hframes()

Vector3 offset = Vector3(0, 0)

The mesh's placing offset.

void set_offset(Vector3 value)

Vector3 get_offset()

float pixel_size = 0.01

The size of one pixel's width on the sprite to scale it in 3D.

void set_pixel_size(float value)

float get_pixel_size()

bool region_enabled = false

If true, the sprite will use region_rect and display only the specified part of its texture.

void set_region_enabled(bool value)

bool is_region_enabled()

Rect2 region_rect = Rect2(0, 0, 0, 0)

The region of the atlas texture to display. region_enabled must be true.

void set_region_rect(Rect2 value)

Rect2 get_region_rect()

Texture2D texture

Texture2D object to draw.

void set_texture (Texture2D value)

Texture2D get_texture()

float uv_correction = 0.0

Sometimes, the UV mapping would leak the color of adjacent pixels into parts of the mesh where they shouldn't be. As a result, some lines of color may appear at the border of some faces.

This property aims to fix this problem. When its value increases, the UV mapping would move inwards. Be careful, as it would also produce misalignment.

void set_uv_correction(float uv_correction)

float get_uv_correction()

int vframes = 1

The number of rows in the sprite sheet.

void set_vframes(int value)

int get_vframes()

Method Descriptions

ArrayMesh get_mesh_with_index(int index)

Returns the mesh that corresponds to a frame of the animation, represented by its index.

void update_sprite_mesh()

Generates the meshes and material given the current properties.

Support

Just give a good ol' star to this GitHub repository if you enjoy this plugin, it means a lot.

Credits

Sprites for the demo project are from Ninja Adventure, made by Pixel-Boy and AAA, check out their work!