From 4da6c31ba99076139c83e429e8ea582947464bca Mon Sep 17 00:00:00 2001 From: kleonc <9283098+kleonc@users.noreply.github.com> Date: Sun, 18 Sep 2022 20:27:22 +0200 Subject: [PATCH] `SpriteBase3D` Fix drawing AtlasTextures with vertical margins differently than in 2D --- scene/3d/sprite_3d.cpp | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/scene/3d/sprite_3d.cpp b/scene/3d/sprite_3d.cpp index 5048aa5e6..ec99ec40d 100644 --- a/scene/3d/sprite_3d.cpp +++ b/scene/3d/sprite_3d.cpp @@ -104,11 +104,28 @@ void SpriteBase3D::draw_texture_rect(Ref p_texture, Rect2 p_dst_rect, R return; } + // 2D: 3D plane (axes match exactly when `axis == Vector3::AXIS_Z`): + // -X+ -X+ + // - + + // Y +--------+ +--------+ +--------+ Y +--------+ + // + | +--+ | | | (2) | | - | 0--1 | + // | |ab| | (1) | +--+ | (3) | 3--2 | | |ab| | + // | |cd| | --> | |ab| | --> | |cd| | <==> | |cd| | + // | +--+ | | |cd| | | |ab| | | 3--2 | + // | | | +--+ | | 0--1 | | | + // +--------+ +--------+ +--------+ +--------+ + + // (1) Y-wise shift `final_rect` within `p_dst_rect` so after inverting Y + // axis distances between top/bottom borders will be preserved (so for + // example AtlasTextures with vertical margins will look the same in 2D/3D). + final_rect.position.y = (p_dst_rect.position.y + p_dst_rect.size.y) - ((final_rect.position.y + final_rect.size.y) - p_dst_rect.position.y); + Color color = _get_color_accum(); color.a *= get_opacity(); float pixel_size = get_pixel_size(); + // (2) Order vertices (0123) bottom-top in 2D / top-bottom in 3D. Vector2 vertices[4] = { (final_rect.position + Vector2(0, final_rect.size.y)) * pixel_size, (final_rect.position + final_rect.size) * pixel_size, @@ -125,6 +142,7 @@ void SpriteBase3D::draw_texture_rect(Ref p_texture, Rect2 p_dst_rect, R src_tsize[1] = atlas_tex->get_atlas()->get_height(); } + // (3) Assign UVs (abcd) according to the vertices order (bottom-top in 2D / top-bottom in 3D). Vector2 uvs[4] = { final_src_rect.position / src_tsize, (final_src_rect.position + Vector2(final_src_rect.size.x, 0)) / src_tsize,