From dc6d8d60982a545fc7ca970468b13799fe700e8e Mon Sep 17 00:00:00 2001 From: bruvzg <7645683+bruvzg@users.noreply.github.com> Date: Sat, 21 May 2022 20:12:12 +0300 Subject: [PATCH] Backport Label3D node implementation and Sprite*3D material render priority. --- doc/classes/Font.xml | 45 + doc/classes/Label3D.xml | 156 ++++ doc/classes/SpatialMaterial.xml | 8 +- doc/classes/SpriteBase3D.xml | 19 +- editor/icons/icon_label_3d.svg | 1 + editor/plugins/spatial_editor_plugin.cpp | 1 + editor/spatial_editor_gizmos.cpp | 33 + editor/spatial_editor_gizmos.h | 13 + scene/3d/label_3d.cpp | 1051 ++++++++++++++++++++++ scene/3d/label_3d.h | 228 +++++ scene/3d/sprite_3d.cpp | 28 +- scene/3d/sprite_3d.h | 6 + scene/gui/label.cpp | 2 +- scene/register_scene_types.cpp | 2 + scene/resources/dynamic_font.cpp | 216 +++++ scene/resources/dynamic_font.h | 14 + scene/resources/font.cpp | 141 +++ scene/resources/font.h | 14 + scene/resources/material.cpp | 53 +- scene/resources/material.h | 7 +- 20 files changed, 2011 insertions(+), 27 deletions(-) create mode 100644 doc/classes/Label3D.xml create mode 100644 editor/icons/icon_label_3d.svg create mode 100644 scene/3d/label_3d.cpp create mode 100644 scene/3d/label_3d.h diff --git a/doc/classes/Font.xml b/doc/classes/Font.xml index 1ca93d4c0fd0..be03a11fe882 100644 --- a/doc/classes/Font.xml +++ b/doc/classes/Font.xml @@ -51,6 +51,51 @@ Returns the size of a character, optionally taking kerning into account if the next character is provided. Note that the height returned is the font height (see [method get_height]) and has no relation to the glyph height. + + + + + + + Returns resource id of the cache texture containing the char. + + + + + + + + + Returns size of the cache texture containing the char. + + + + + + + + + Returns char offset from the baseline. + + + + + + + + + Returns size of the char. + + + + + + + + + Returns rectangle in the cache texture containing the char. + + diff --git a/doc/classes/Label3D.xml b/doc/classes/Label3D.xml new file mode 100644 index 000000000000..6446ea1cf6ef --- /dev/null +++ b/doc/classes/Label3D.xml @@ -0,0 +1,156 @@ + + + + Displays plain text in a 3D world. + + + Label3D displays plain text in a 3D world. It gives you control over the horizontal and vertical alignment. + + + + + + + + Returns a [TriangleMesh] with the label's vertices following its current configuration (such as its [member pixel_size]). + + + + + + + Returns the value of the specified flag. + + + + + + + + If [code]true[/code], the specified flag will be enabled. See [enum Label3D.DrawFlags] for a list of flags. + + + + + + The alpha cutting mode to use for the sprite. See [enum AlphaCutMode] for possible values. + + + Threshold at which the alpha scissor will discard values. + + + If [code]true[/code], wraps the text to the [member width]. + + + The billboard mode to use for the label. See [enum SpatialMaterial.BillboardMode] for possible values. + + + If [code]true[/code], text can be seen from the back as well, if [code]false[/code], it is invisible when looking at it from behind. + + + If [code]true[/code], the label is rendered at the same size regardless of distance. + + + [Font] used for the [Label3D]'s text. + + + Controls the text's horizontal alignment. Supports left, center, right. Set it to one of the [enum Align] constants. + + + Vertical space between lines in multiline [Label3D]. + + + Text [Color] of the [Label3D]. + + + If [code]true[/code], depth testing is disabled and the object will be drawn in render order. + + + The text drawing offset (in pixels). + + + The tint of [Font]'s outline. + + + Sets the render priority for the text outline. Higher priority objects will be sorted in front of lower priority objects. + [b]Node:[/b] This only applies if [member alpha_cut] is set to [constant ALPHA_CUT_DISABLED] (default value). + [b]Note:[/b] This only applies to sorting of transparent objects. This will not impact how transparent objects are sorted relative to opaque objects. This is because opaque objects are not sorted, while transparent objects are sorted from back to front (subject to priority). + + + The size of one pixel's width on the label to scale it in 3D. + + + Sets the render priority for the text. Higher priority objects will be sorted in front of lower priority objects. + [b]Node:[/b] This only applies if [member alpha_cut] is set to [constant ALPHA_CUT_DISABLED] (default value). + [b]Note:[/b] This only applies to sorting of transparent objects. This will not impact how transparent objects are sorted relative to opaque objects. This is because opaque objects are not sorted, while transparent objects are sorted from back to front (subject to priority). + + + If [code]true[/code], the [Light] in the [Environment] has effects on the label. + + + The text to display on screen. + + + If [code]true[/code], all the text displays as UPPERCASE. + + + Controls the text's vertical alignment. Supports top, center, bottom. Set it to one of the [enum VAlign] constants. + + + Text width (in pixels), used for autowrap and fill alignment. + + + + + If set, lights in the environment affect the label. + + + If set, text can be seen from the back as well. If not, the texture is invisible when looking at it from behind. + + + Disables the depth test, so this object is drawn on top of all others. However, objects drawn after it in the draw order may cover it. + + + Label is scaled by depth so that it always appears the same size on screen. + + + Represents the size of the [enum DrawFlags] enum. + + + This mode performs standard alpha blending. It can display translucent areas, but transparency sorting issues may be visible when multiple transparent materials are overlapping. + + + This mode only allows fully transparent or fully opaque pixels. This mode is also known as [i]alpha testing[/i] or [i]1-bit transparency[/i]. + [b]Note:[/b] This mode might have issues with anti-aliased fonts and outlines, try adjusting [member alpha_scissor_threshold] or using SDF font. + [b]Note:[/b] When using text with overlapping glyphs (e.g., cursive scripts), this mode might have transparency sorting issues between the main text and the outline. + + + This mode draws fully opaque pixels in the depth prepass. This is slower than [constant ALPHA_CUT_DISABLED] or [constant ALPHA_CUT_DISCARD], but it allows displaying translucent areas and smooth edges while using proper sorting. + [b]Note:[/b] When using text with overlapping glyphs (e.g., cursive scripts), this mode might have transparency sorting issues between the main text and the outline. + + + Align rows to the left (default). + + + Align rows centered. + + + Align rows to the right. + + + Expand row whitespaces to fit the width. + + + Align the whole text to the top. + + + Align the whole text to the center. + + + Align the whole text to the bottom. + + + Align the whole text by spreading the rows. + + + diff --git a/doc/classes/SpatialMaterial.xml b/doc/classes/SpatialMaterial.xml index db74b7d914b6..5fd1c381f71f 100644 --- a/doc/classes/SpatialMaterial.xml +++ b/doc/classes/SpatialMaterial.xml @@ -183,6 +183,9 @@ Forces a conversion of the [member albedo_texture] from sRGB space to linear space. + + Enables signed distance field rendering shader. + If [code]true[/code], the object receives no ambient light. @@ -570,7 +573,10 @@ Enables the shadow to opacity feature. - + + Enables signed distance field rendering shader. + + Represents the size of the [enum Flags] enum. diff --git a/doc/classes/SpriteBase3D.xml b/doc/classes/SpriteBase3D.xml index f2c77d69ba69..57c27ca8be10 100644 --- a/doc/classes/SpriteBase3D.xml +++ b/doc/classes/SpriteBase3D.xml @@ -50,6 +50,9 @@ If [code]true[/code], texture can be seen from the back as well, if [code]false[/code], it is invisible when looking at it from behind. + + If [code]true[/code], the label is rendered at the same size regardless of distance. + If [code]true[/code], texture is flipped horizontally. @@ -60,6 +63,9 @@ A color value used to [i]multiply[/i] the texture's colors. Can be used for mood-coloring or to simulate the color of light. [b]Note:[/b] If a [member GeometryInstance.material_override] is defined on the [SpriteBase3D], the material override must be configured to take vertex colors into account for albedo. Otherwise, the color defined in [member modulate] will be ignored. For a [SpatialMaterial], [member SpatialMaterial.vertex_color_use_as_albedo] must be [code]true[/code]. For a [ShaderMaterial], [code]ALBEDO *= COLOR.rgb;[/code] must be inserted in the shader's [code]fragment()[/code] function. + + If [code]true[/code], depth testing is disabled and the object will be drawn in render order. + The texture's drawing offset. @@ -70,6 +76,11 @@ The size of one pixel's width on the sprite to scale it in 3D. + + Sets the render priority for the sprite. Higher priority objects will be sorted in front of lower priority objects. + [b]Node:[/b] This only applies if [member alpha_cut] is set to [constant ALPHA_CUT_DISABLED] (default value). + [b]Note:[/b] This only applies to sorting of transparent objects. This will not impact how transparent objects are sorted relative to opaque objects. This is because opaque objects are not sorted, while transparent objects are sorted from back to front (subject to priority). + If [code]true[/code], the [Light] in the [Environment] has effects on the sprite. @@ -87,7 +98,13 @@ If set, texture can be seen from the back as well, if not, it is invisible when looking at it from behind. - + + Disables the depth test, so this object is drawn on top of all others. However, objects drawn after it in the draw order may cover it. + + + Sprite is scaled by depth so that it always appears the same size on screen. + + Represents the size of the [enum DrawFlags] enum. diff --git a/editor/icons/icon_label_3d.svg b/editor/icons/icon_label_3d.svg new file mode 100644 index 000000000000..76e1b7c276d4 --- /dev/null +++ b/editor/icons/icon_label_3d.svg @@ -0,0 +1 @@ + diff --git a/editor/plugins/spatial_editor_plugin.cpp b/editor/plugins/spatial_editor_plugin.cpp index a2af4b330ad7..9943af0a08e1 100644 --- a/editor/plugins/spatial_editor_plugin.cpp +++ b/editor/plugins/spatial_editor_plugin.cpp @@ -6488,6 +6488,7 @@ void SpatialEditor::_register_all_gizmos() { add_gizmo_plugin(Ref(memnew(MeshInstanceSpatialGizmoPlugin))); add_gizmo_plugin(Ref(memnew(SoftBodySpatialGizmoPlugin))); add_gizmo_plugin(Ref(memnew(Sprite3DSpatialGizmoPlugin))); + add_gizmo_plugin(Ref(memnew(Label3DSpatialGizmoPlugin))); add_gizmo_plugin(Ref(memnew(SkeletonSpatialGizmoPlugin))); add_gizmo_plugin(Ref(memnew(Position3DSpatialGizmoPlugin))); add_gizmo_plugin(Ref(memnew(RayCastSpatialGizmoPlugin))); diff --git a/editor/spatial_editor_gizmos.cpp b/editor/spatial_editor_gizmos.cpp index 30b8fa30c80e..205b5310cd37 100644 --- a/editor/spatial_editor_gizmos.cpp +++ b/editor/spatial_editor_gizmos.cpp @@ -38,6 +38,7 @@ #include "scene/3d/collision_shape.h" #include "scene/3d/cpu_particles.h" #include "scene/3d/gi_probe.h" +#include "scene/3d/label_3d.h" #include "scene/3d/light.h" #include "scene/3d/listener.h" #include "scene/3d/mesh_instance.h" @@ -1563,6 +1564,38 @@ void Sprite3DSpatialGizmoPlugin::redraw(EditorSpatialGizmo *p_gizmo) { /// +Label3DSpatialGizmoPlugin::Label3DSpatialGizmoPlugin() { +} + +bool Label3DSpatialGizmoPlugin::has_gizmo(Spatial *p_spatial) { + return Object::cast_to(p_spatial) != nullptr; +} + +String Label3DSpatialGizmoPlugin::get_name() const { + return "Label3D"; +} + +int Label3DSpatialGizmoPlugin::get_priority() const { + return -1; +} + +bool Label3DSpatialGizmoPlugin::can_be_hidden() const { + return false; +} + +void Label3DSpatialGizmoPlugin::redraw(EditorSpatialGizmo *p_gizmo) { + Label3D *label = Object::cast_to(p_gizmo->get_spatial_node()); + + p_gizmo->clear(); + + Ref tm = label->generate_triangle_mesh(); + if (tm.is_valid()) { + p_gizmo->add_collision_triangles(tm); + } +} + +/// + Position3DSpatialGizmoPlugin::Position3DSpatialGizmoPlugin() { pos3d_mesh = Ref(memnew(ArrayMesh)); cursor_points = Vector(); diff --git a/editor/spatial_editor_gizmos.h b/editor/spatial_editor_gizmos.h index 0eeef772cc6e..db3788c8045b 100644 --- a/editor/spatial_editor_gizmos.h +++ b/editor/spatial_editor_gizmos.h @@ -126,6 +126,19 @@ class Sprite3DSpatialGizmoPlugin : public EditorSpatialGizmoPlugin { Sprite3DSpatialGizmoPlugin(); }; +class Label3DSpatialGizmoPlugin : public EditorSpatialGizmoPlugin { + GDCLASS(Label3DSpatialGizmoPlugin, EditorSpatialGizmoPlugin); + +public: + bool has_gizmo(Spatial *p_spatial); + String get_name() const; + int get_priority() const; + bool can_be_hidden() const; + void redraw(EditorSpatialGizmo *p_gizmo); + + Label3DSpatialGizmoPlugin(); +}; + class Position3DSpatialGizmoPlugin : public EditorSpatialGizmoPlugin { GDCLASS(Position3DSpatialGizmoPlugin, EditorSpatialGizmoPlugin); diff --git a/scene/3d/label_3d.cpp b/scene/3d/label_3d.cpp new file mode 100644 index 000000000000..4a2c50e3d644 --- /dev/null +++ b/scene/3d/label_3d.cpp @@ -0,0 +1,1051 @@ +/*************************************************************************/ +/* label_3d.cpp */ +/*************************************************************************/ +/* This file is part of: */ +/* GODOT ENGINE */ +/* https://godotengine.org */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */ +/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */ +/* */ +/* Permission is hereby granted, free of charge, to any person obtaining */ +/* a copy of this software and associated documentation files (the */ +/* "Software"), to deal in the Software without restriction, including */ +/* without limitation the rights to use, copy, modify, merge, publish, */ +/* distribute, sublicense, and/or sell copies of the Software, and to */ +/* permit persons to whom the Software is furnished to do so, subject to */ +/* the following conditions: */ +/* */ +/* The above copyright notice and this permission notice shall be */ +/* included in all copies or substantial portions of the Software. */ +/* */ +/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ +/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ +/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ +/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ +/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ +/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ +/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ +/*************************************************************************/ + +#include "label_3d.h" + +#include "core/core_string_names.h" +#include "scene/resources/theme.h" +#include "scene/scene_string_names.h" + +void Label3D::_bind_methods() { + ClassDB::bind_method(D_METHOD("set_horizontal_alignment", "alignment"), &Label3D::set_horizontal_alignment); + ClassDB::bind_method(D_METHOD("get_horizontal_alignment"), &Label3D::get_horizontal_alignment); + + ClassDB::bind_method(D_METHOD("set_vertical_alignment", "alignment"), &Label3D::set_vertical_alignment); + ClassDB::bind_method(D_METHOD("get_vertical_alignment"), &Label3D::get_vertical_alignment); + + ClassDB::bind_method(D_METHOD("set_modulate", "modulate"), &Label3D::set_modulate); + ClassDB::bind_method(D_METHOD("get_modulate"), &Label3D::get_modulate); + + ClassDB::bind_method(D_METHOD("set_outline_modulate", "modulate"), &Label3D::set_outline_modulate); + ClassDB::bind_method(D_METHOD("get_outline_modulate"), &Label3D::get_outline_modulate); + + ClassDB::bind_method(D_METHOD("set_text", "text"), &Label3D::set_text); + ClassDB::bind_method(D_METHOD("get_text"), &Label3D::get_text); + + ClassDB::bind_method(D_METHOD("set_uppercase", "enable"), &Label3D::set_uppercase); + ClassDB::bind_method(D_METHOD("is_uppercase"), &Label3D::is_uppercase); + + ClassDB::bind_method(D_METHOD("set_render_priority", "priority"), &Label3D::set_render_priority); + ClassDB::bind_method(D_METHOD("get_render_priority"), &Label3D::get_render_priority); + + ClassDB::bind_method(D_METHOD("set_outline_render_priority", "priority"), &Label3D::set_outline_render_priority); + ClassDB::bind_method(D_METHOD("get_outline_render_priority"), &Label3D::get_outline_render_priority); + + ClassDB::bind_method(D_METHOD("set_font", "font"), &Label3D::set_font); + ClassDB::bind_method(D_METHOD("get_font"), &Label3D::get_font); + + ClassDB::bind_method(D_METHOD("set_line_spacing", "line_spacing"), &Label3D::set_line_spacing); + ClassDB::bind_method(D_METHOD("get_line_spacing"), &Label3D::get_line_spacing); + + ClassDB::bind_method(D_METHOD("set_autowrap", "autowrap_mode"), &Label3D::set_autowrap); + ClassDB::bind_method(D_METHOD("get_autowrap"), &Label3D::get_autowrap); + + ClassDB::bind_method(D_METHOD("set_width", "width"), &Label3D::set_width); + ClassDB::bind_method(D_METHOD("get_width"), &Label3D::get_width); + + ClassDB::bind_method(D_METHOD("set_pixel_size", "pixel_size"), &Label3D::set_pixel_size); + ClassDB::bind_method(D_METHOD("get_pixel_size"), &Label3D::get_pixel_size); + + ClassDB::bind_method(D_METHOD("set_offset", "offset"), &Label3D::set_offset); + ClassDB::bind_method(D_METHOD("get_offset"), &Label3D::get_offset); + + ClassDB::bind_method(D_METHOD("set_draw_flag", "flag", "enabled"), &Label3D::set_draw_flag); + ClassDB::bind_method(D_METHOD("get_draw_flag", "flag"), &Label3D::get_draw_flag); + + ClassDB::bind_method(D_METHOD("set_billboard_mode", "mode"), &Label3D::set_billboard_mode); + ClassDB::bind_method(D_METHOD("get_billboard_mode"), &Label3D::get_billboard_mode); + + ClassDB::bind_method(D_METHOD("set_alpha_cut_mode", "mode"), &Label3D::set_alpha_cut_mode); + ClassDB::bind_method(D_METHOD("get_alpha_cut_mode"), &Label3D::get_alpha_cut_mode); + + ClassDB::bind_method(D_METHOD("set_alpha_scissor_threshold", "threshold"), &Label3D::set_alpha_scissor_threshold); + ClassDB::bind_method(D_METHOD("get_alpha_scissor_threshold"), &Label3D::get_alpha_scissor_threshold); + + ClassDB::bind_method(D_METHOD("generate_triangle_mesh"), &Label3D::generate_triangle_mesh); + + ClassDB::bind_method(D_METHOD("_queue_update"), &Label3D::_queue_update); + ClassDB::bind_method(D_METHOD("_font_changed"), &Label3D::_font_changed); + ClassDB::bind_method(D_METHOD("_im_update"), &Label3D::_im_update); + + ADD_PROPERTY(PropertyInfo(Variant::REAL, "pixel_size", PROPERTY_HINT_RANGE, "0.0001,128,0.0001"), "set_pixel_size", "get_pixel_size"); + ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "offset"), "set_offset", "get_offset"); + + ADD_GROUP("Flags", ""); + ADD_PROPERTY(PropertyInfo(Variant::INT, "billboard", PROPERTY_HINT_ENUM, "Disabled,Enabled,Y-Billboard"), "set_billboard_mode", "get_billboard_mode"); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "shaded"), "set_draw_flag", "get_draw_flag", FLAG_SHADED); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "double_sided"), "set_draw_flag", "get_draw_flag", FLAG_DOUBLE_SIDED); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "no_depth_test"), "set_draw_flag", "get_draw_flag", FLAG_DISABLE_DEPTH_TEST); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "fixed_size"), "set_draw_flag", "get_draw_flag", FLAG_FIXED_SIZE); + ADD_PROPERTY(PropertyInfo(Variant::INT, "alpha_cut", PROPERTY_HINT_ENUM, "Disabled,Discard,Opaque Pre-Pass"), "set_alpha_cut_mode", "get_alpha_cut_mode"); + ADD_PROPERTY(PropertyInfo(Variant::REAL, "alpha_scissor_threshold", PROPERTY_HINT_RANGE, "0,1,0.01"), "set_alpha_scissor_threshold", "get_alpha_scissor_threshold"); + ADD_PROPERTY(PropertyInfo(Variant::INT, "render_priority", PROPERTY_HINT_RANGE, itos(VS::MATERIAL_RENDER_PRIORITY_MIN) + "," + itos(VS::MATERIAL_RENDER_PRIORITY_MAX) + ",1"), "set_render_priority", "get_render_priority"); + ADD_PROPERTY(PropertyInfo(Variant::INT, "outline_render_priority", PROPERTY_HINT_RANGE, itos(VS::MATERIAL_RENDER_PRIORITY_MIN) + "," + itos(VS::MATERIAL_RENDER_PRIORITY_MAX) + ",1"), "set_outline_render_priority", "get_outline_render_priority"); + + ADD_GROUP("Text", ""); + ADD_PROPERTY(PropertyInfo(Variant::COLOR, "modulate"), "set_modulate", "get_modulate"); + ADD_PROPERTY(PropertyInfo(Variant::COLOR, "outline_modulate"), "set_outline_modulate", "get_outline_modulate"); + ADD_PROPERTY(PropertyInfo(Variant::STRING, "text", PROPERTY_HINT_MULTILINE_TEXT, ""), "set_text", "get_text"); + ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "font", PROPERTY_HINT_RESOURCE_TYPE, "Font"), "set_font", "get_font"); + ADD_PROPERTY(PropertyInfo(Variant::INT, "horizontal_alignment", PROPERTY_HINT_ENUM, "Left,Center,Right,Fill"), "set_horizontal_alignment", "get_horizontal_alignment"); + ADD_PROPERTY(PropertyInfo(Variant::INT, "vertical_alignment", PROPERTY_HINT_ENUM, "Top,Center,Bottom"), "set_vertical_alignment", "get_vertical_alignment"); + ADD_PROPERTY(PropertyInfo(Variant::BOOL, "uppercase"), "set_uppercase", "is_uppercase"); + ADD_PROPERTY(PropertyInfo(Variant::REAL, "line_spacing"), "set_line_spacing", "get_line_spacing"); + ADD_PROPERTY(PropertyInfo(Variant::BOOL, "autowrap"), "set_autowrap", "get_autowrap"); + ADD_PROPERTY(PropertyInfo(Variant::REAL, "width"), "set_width", "get_width"); + + BIND_ENUM_CONSTANT(FLAG_SHADED); + BIND_ENUM_CONSTANT(FLAG_DOUBLE_SIDED); + BIND_ENUM_CONSTANT(FLAG_DISABLE_DEPTH_TEST); + BIND_ENUM_CONSTANT(FLAG_FIXED_SIZE); + BIND_ENUM_CONSTANT(FLAG_MAX); + + BIND_ENUM_CONSTANT(ALPHA_CUT_DISABLED); + BIND_ENUM_CONSTANT(ALPHA_CUT_DISCARD); + BIND_ENUM_CONSTANT(ALPHA_CUT_OPAQUE_PREPASS); + + BIND_ENUM_CONSTANT(ALIGN_LEFT); + BIND_ENUM_CONSTANT(ALIGN_CENTER); + BIND_ENUM_CONSTANT(ALIGN_RIGHT); + BIND_ENUM_CONSTANT(ALIGN_FILL); + + BIND_ENUM_CONSTANT(VALIGN_TOP); + BIND_ENUM_CONSTANT(VALIGN_CENTER); + BIND_ENUM_CONSTANT(VALIGN_BOTTOM); + BIND_ENUM_CONSTANT(VALIGN_FILL); +} + +void Label3D::_validate_property(PropertyInfo &property) const { + if (property.name == "material_override" || property.name == "material_overlay") { + property.usage = PROPERTY_USAGE_NOEDITOR; + } +} + +int Label3D::get_longest_line_width() const { + Ref font = _get_font_or_default(); + real_t max_line_width = 0; + real_t line_width = 0; + + for (int i = 0; i < xl_text.size(); i++) { + CharType current = xl_text[i]; + if (uppercase) { + current = String::char_uppercase(current); + } + + if (current < 32) { + if (current == '\n') { + if (line_width > max_line_width) { + max_line_width = line_width; + } + line_width = 0; + } + } else { + real_t char_width = font->get_char_size(current, xl_text[i + 1]).width; + line_width += char_width; + } + } + + if (line_width > max_line_width) { + max_line_width = line_width; + } + + // ceiling to ensure autowrapping does not cut text + return Math::ceil(max_line_width); +} + +void Label3D::regenerate_word_cache() { + while (word_cache) { + WordCache *current = word_cache; + word_cache = current->next; + memdelete(current); + } + + int max_line_width; + if (autowrap) { + max_line_width = width; + } else { + max_line_width = get_longest_line_width(); + } + + Ref font = _get_font_or_default(); + + real_t current_word_size = 0; + int word_pos = 0; + real_t line_width = 0; + int space_count = 0; + real_t space_width = font->get_char_size(' ').width; + line_count = 1; + bool was_separatable = false; + + WordCache *last = nullptr; + + for (int i = 0; i <= xl_text.length(); i++) { + CharType current = i < xl_text.length() ? xl_text[i] : L' '; //always a space at the end, so the algo works + + if (uppercase) { + current = String::char_uppercase(current); + } + + // ranges taken from https://en.wikipedia.org/wiki/Plane_(Unicode) + // if your language is not well supported, consider helping improve + // the unicode support in Godot. + bool separatable = (current >= 0x2E08 && current <= 0x9FFF) || // CJK scripts and symbols. + (current >= 0xAC00 && current <= 0xD7FF) || // Hangul Syllables and Hangul Jamo Extended-B. + (current >= 0xF900 && current <= 0xFAFF) || // CJK Compatibility Ideographs. + (current >= 0xFE30 && current <= 0xFE4F) || // CJK Compatibility Forms. + (current >= 0xFF65 && current <= 0xFF9F) || // Halfwidth forms of katakana + (current >= 0xFFA0 && current <= 0xFFDC) || // Halfwidth forms of compatibility jamo characters for Hangul + (current >= 0x20000 && current <= 0x2FA1F) || // CJK Unified Ideographs Extension B ~ F and CJK Compatibility Ideographs Supplement. + (current >= 0x30000 && current <= 0x3134F); // CJK Unified Ideographs Extension G. + bool insert_newline = false; + real_t char_width = 0; + + bool separation_changed = i > 0 && was_separatable != separatable; + was_separatable = separatable; + + if (current < 33) { // Control characters and space. + if (current_word_size > 0) { // These characters always create a word-break. + WordCache *wc = memnew(WordCache); + if (word_cache) { + last->next = wc; + } else { + word_cache = wc; + } + last = wc; + + wc->pixel_width = current_word_size; + wc->char_pos = word_pos; + wc->word_len = i - word_pos; + wc->space_count = space_count; + current_word_size = 0; + space_count = 0; + } else if ((i == xl_text.length() || current == '\n') && last != nullptr && space_count != 0) { + // In case there are trailing white spaces we add a placeholder word cache with just the spaces. + WordCache *wc = memnew(WordCache); + if (word_cache) { + last->next = wc; + } else { + word_cache = wc; + } + last = wc; + + wc->pixel_width = 0; + wc->char_pos = 0; + wc->word_len = 0; + wc->space_count = space_count; + current_word_size = 0; + space_count = 0; + } + + if (current == '\n') { + insert_newline = true; + } + + if (i < xl_text.length() && xl_text[i] == ' ') { + if (line_width == 0) { + if (current_word_size == 0) { + word_pos = i; + } + current_word_size += space_width; + line_width += space_width; + } else if (line_width > 0 || last == nullptr || last->char_pos != WordCache::CHAR_WRAPLINE) { + space_count++; + line_width += space_width; + } else { + space_count = 0; + } + } + + } else { // Characters with graphical representation. + // Word-break on CJK & non-CJK edge. + if (separation_changed && current_word_size > 0) { + WordCache *wc = memnew(WordCache); + if (word_cache) { + last->next = wc; + } else { + word_cache = wc; + } + last = wc; + + wc->pixel_width = current_word_size; + wc->char_pos = word_pos; + wc->word_len = i - word_pos; + wc->space_count = space_count; + current_word_size = 0; + space_count = 0; + } + if (current_word_size == 0) { + word_pos = i; + } + char_width = font->get_char_size(current, xl_text[i + 1]).width; + current_word_size += char_width; + line_width += char_width; + + // allow autowrap to cut words when they exceed line width + if (autowrap && (current_word_size > max_line_width)) { + separatable = true; + } + } + + if ((autowrap && (line_width >= max_line_width) && ((last && last->char_pos >= 0) || separatable)) || insert_newline) { + if (separatable) { + if (current_word_size > 0) { + WordCache *wc = memnew(WordCache); + if (word_cache) { + last->next = wc; + } else { + word_cache = wc; + } + last = wc; + + wc->pixel_width = current_word_size - char_width; + wc->char_pos = word_pos; + wc->word_len = i - word_pos; + wc->space_count = space_count; + current_word_size = char_width; + word_pos = i; + } + } + + WordCache *wc = memnew(WordCache); + if (word_cache) { + last->next = wc; + } else { + word_cache = wc; + } + last = wc; + + wc->pixel_width = 0; + wc->char_pos = insert_newline ? WordCache::CHAR_NEWLINE : WordCache::CHAR_WRAPLINE; + + line_width = current_word_size; + line_count++; + space_count = 0; + } + } + + word_cache_dirty = false; +} + +void Label3D::_notification(int p_what) { + switch (p_what) { + case NOTIFICATION_ENTER_TREE: { + if (!pending_update) { + _im_update(); + } + } break; + case NOTIFICATION_TRANSLATION_CHANGED: { + String new_text = tr(text); + if (new_text == xl_text) { + return; // Nothing new. + } + xl_text = new_text; + + regenerate_word_cache(); + _queue_update(); + } break; + } +} + +void Label3D::_im_update() { + _shape(); + + triangle_mesh.unref(); + update_gizmo(); + + pending_update = false; +} + +void Label3D::_queue_update() { + if (pending_update) { + return; + } + + pending_update = true; + call_deferred(SceneStringNames::get_singleton()->_im_update); +} + +AABB Label3D::get_aabb() const { + return aabb; +} + +Ref Label3D::generate_triangle_mesh() const { + if (triangle_mesh.is_valid()) { + return triangle_mesh; + } + + Ref font = _get_font_or_default(); + if (font.is_null()) { + return Ref(); + } + + PoolVector faces; + faces.resize(6); + PoolVector::Write facesw = faces.write(); + + if (word_cache_dirty) { + const_cast(this)->regenerate_word_cache(); + } + + int font_h = font->get_height() + line_spacing; + real_t space_w = font->get_char_size(' ').width; + float total_h = line_count * font_h; + + float vbegin = 0; + switch (vertical_alignment) { + case VALIGN_FILL: + case VALIGN_TOP: { + // Nothing. + } break; + case VALIGN_CENTER: { + vbegin = (total_h - line_spacing) / 2.0; + } break; + case VALIGN_BOTTOM: { + vbegin = (total_h - line_spacing); + } break; + } + + WordCache *wc = word_cache; + if (!wc) { + return Ref(); + } + + float max_line_w = 0.0; + int line = 0; + while (wc) { + if (line >= line_count) { + break; + } + + if (wc->char_pos < 0) { + wc = wc->next; + line++; + continue; + } + + WordCache *to = wc; + + float taken = 0; + int spaces = 0; + while (to && to->char_pos >= 0) { + taken += to->pixel_width; + if (to->space_count) { + spaces += to->space_count; + } + to = to->next; + } + + max_line_w = MAX(max_line_w, (taken + spaces * space_w)); + + wc = to ? to->next : nullptr; + line++; + } + + Vector2 offset = Vector2(0, vbegin); + switch (horizontal_alignment) { + case ALIGN_FILL: + case ALIGN_LEFT: { + // Noting + } break; + case ALIGN_CENTER: { + offset.x = -max_line_w / 2.0; + } break; + case ALIGN_RIGHT: { + offset.x = -max_line_w; + } break; + } + + Rect2 final_rect = Rect2(offset + lbl_offset, Size2(max_line_w, total_h)); + + if (final_rect.size.x == 0 || final_rect.size.y == 0) { + return Ref(); + } + + real_t pixel_size = get_pixel_size(); + + Vector2 vertices[4] = { + + (final_rect.position + Vector2(0, -final_rect.size.y)) * pixel_size, + (final_rect.position + Vector2(final_rect.size.x, -final_rect.size.y)) * pixel_size, + (final_rect.position + Vector2(final_rect.size.x, 0)) * pixel_size, + final_rect.position * pixel_size, + + }; + + static const int indices[6] = { + 0, 1, 2, + 0, 2, 3 + }; + + for (int j = 0; j < 6; j++) { + int i = indices[j]; + Vector3 vtx; + vtx[0] = vertices[i][0]; + vtx[1] = vertices[i][1]; + facesw[j] = vtx; + } + + triangle_mesh = Ref(memnew(TriangleMesh)); + triangle_mesh->create(faces); + + return triangle_mesh; +} + +PoolVector Label3D::get_faces(uint32_t p_usage_flags) const { + return PoolVector(); +} + +float Label3D::_generate_glyph_surfaces(const Ref &p_font, CharType p_char, CharType p_next, Vector2 p_offset, const Color &p_modulate, int p_priority, bool p_outline) { + Vector2 gl_of; + Vector2 gl_sz; + Rect2 gl_uv; + Size2 texs; + RID tex; + + tex = p_font->get_char_texture(p_char, p_next, p_outline); + gl_of = p_font->get_char_tx_offset(p_char, p_next, p_outline); + gl_sz = p_font->get_char_tx_size(p_char, p_next, p_outline); + gl_uv = p_font->get_char_tx_uv_rect(p_char, p_next, p_outline); + texs = p_font->get_char_texture_size(p_char, p_next, p_outline); + + uint64_t mat_hash; + if (tex != RID()) { + mat_hash = hash_one_uint64(tex.get_id()); + } else { + mat_hash = hash_one_uint64(0); + } + mat_hash = hash_djb2_one_64(p_priority, mat_hash); + + if (!surfaces.has(mat_hash)) { + SurfaceData surf; + surf.material = RID_PRIME(VisualServer::get_singleton()->material_create()); + // Set defaults for material, names need to match up those in SpatialMaterial + VS::get_singleton()->material_set_param(surf.material, "albedo", Color(1, 1, 1, 1)); + VS::get_singleton()->material_set_param(surf.material, "specular", 0.5); + VS::get_singleton()->material_set_param(surf.material, "metallic", 0.0); + VS::get_singleton()->material_set_param(surf.material, "roughness", 1.0); + VS::get_singleton()->material_set_param(surf.material, "uv1_offset", Vector3(0, 0, 0)); + VS::get_singleton()->material_set_param(surf.material, "uv1_scale", Vector3(1, 1, 1)); + VS::get_singleton()->material_set_param(surf.material, "uv2_offset", Vector3(0, 0, 0)); + VS::get_singleton()->material_set_param(surf.material, "uv2_scale", Vector3(1, 1, 1)); + VS::get_singleton()->material_set_param(surf.material, "alpha_scissor_threshold", alpha_scissor_threshold); + + RID shader_rid = SpatialMaterial::get_material_rid_for_2d(get_draw_flag(FLAG_SHADED), true, get_draw_flag(FLAG_DOUBLE_SIDED), get_alpha_cut_mode() == ALPHA_CUT_DISCARD, get_alpha_cut_mode() == ALPHA_CUT_OPAQUE_PREPASS, get_billboard_mode() == SpatialMaterial::BILLBOARD_ENABLED, get_billboard_mode() == SpatialMaterial::BILLBOARD_FIXED_Y, get_draw_flag(FLAG_DISABLE_DEPTH_TEST), get_draw_flag(FLAG_FIXED_SIZE), p_font->is_distance_field_hint()); + + VS::get_singleton()->material_set_shader(surf.material, VS::get_singleton()->material_get_shader(shader_rid)); + VS::get_singleton()->material_set_param(surf.material, "texture_albedo", tex); + if (get_alpha_cut_mode() == ALPHA_CUT_DISABLED) { + VS::get_singleton()->material_set_render_priority(surf.material, p_priority); + } else { + surf.z_shift = p_priority; + } + + surfaces[mat_hash] = surf; + } + SurfaceData &s = surfaces[mat_hash]; + + s.mesh_vertices.resize((s.offset + 1) * 4); + s.mesh_normals.resize((s.offset + 1) * 4); + s.mesh_tangents.resize((s.offset + 1) * 16); + s.mesh_colors.resize((s.offset + 1) * 4); + s.mesh_uvs.resize((s.offset + 1) * 4); + + s.mesh_vertices.write()[(s.offset * 4) + 3] = Vector3(p_offset.x + gl_of.x, p_offset.y - gl_of.y - gl_sz.y, s.z_shift) * pixel_size; + s.mesh_vertices.write()[(s.offset * 4) + 2] = Vector3(p_offset.x + gl_of.x + gl_sz.x, p_offset.y - gl_of.y - gl_sz.y, s.z_shift) * pixel_size; + s.mesh_vertices.write()[(s.offset * 4) + 1] = Vector3(p_offset.x + gl_of.x + gl_sz.x, p_offset.y - gl_of.y, s.z_shift) * pixel_size; + s.mesh_vertices.write()[(s.offset * 4) + 0] = Vector3(p_offset.x + gl_of.x, p_offset.y - gl_of.y, s.z_shift) * pixel_size; + + for (int i = 0; i < 4; i++) { + s.mesh_normals.write()[(s.offset * 4) + i] = Vector3(0.0, 0.0, 1.0); + s.mesh_tangents.write()[(s.offset * 16) + (i * 4) + 0] = 1.0; + s.mesh_tangents.write()[(s.offset * 16) + (i * 4) + 1] = 0.0; + s.mesh_tangents.write()[(s.offset * 16) + (i * 4) + 2] = 0.0; + s.mesh_tangents.write()[(s.offset * 16) + (i * 4) + 3] = 1.0; + s.mesh_colors.write()[(s.offset * 4) + i] = p_modulate; + s.mesh_uvs.write()[(s.offset * 4) + i] = Vector2(); + + if (aabb == AABB()) { + aabb.position = s.mesh_vertices[(s.offset * 4) + i]; + } else { + aabb.expand_to(s.mesh_vertices[(s.offset * 4) + i]); + } + } + + if (tex != RID()) { + s.mesh_uvs.write()[(s.offset * 4) + 3] = Vector2(gl_uv.position.x / texs.x, (gl_uv.position.y + gl_uv.size.y) / texs.y); + s.mesh_uvs.write()[(s.offset * 4) + 2] = Vector2((gl_uv.position.x + gl_uv.size.x) / texs.x, (gl_uv.position.y + gl_uv.size.y) / texs.y); + s.mesh_uvs.write()[(s.offset * 4) + 1] = Vector2((gl_uv.position.x + gl_uv.size.x) / texs.x, gl_uv.position.y / texs.y); + s.mesh_uvs.write()[(s.offset * 4) + 0] = Vector2(gl_uv.position.x / texs.x, gl_uv.position.y / texs.y); + } + + s.indices.resize((s.offset + 1) * 6); + s.indices.write()[(s.offset * 6) + 0] = (s.offset * 4) + 0; + s.indices.write()[(s.offset * 6) + 1] = (s.offset * 4) + 1; + s.indices.write()[(s.offset * 6) + 2] = (s.offset * 4) + 2; + s.indices.write()[(s.offset * 6) + 3] = (s.offset * 4) + 0; + s.indices.write()[(s.offset * 6) + 4] = (s.offset * 4) + 2; + s.indices.write()[(s.offset * 6) + 5] = (s.offset * 4) + 3; + + s.offset++; + return p_font->get_char_size(p_char, p_next).x; +} + +void Label3D::_shape() { + // Clear mesh. + VS::get_singleton()->mesh_clear(mesh); + aabb = AABB(); + + // Clear materials. + { + const uint64_t *k = nullptr; + while ((k = surfaces.next(k))) { + VS::get_singleton()->free(surfaces[*k].material); + } + surfaces.clear(); + } + + Ref font = _get_font_or_default(); + ERR_FAIL_COND(font.is_null()); + + if (word_cache_dirty) { + regenerate_word_cache(); + } + + // Generate surfaces and materials. + + int font_h = font->get_height() + line_spacing; + real_t space_w = font->get_char_size(' ').width; + float total_h = line_count * font_h; + + float vbegin = 0.0; + switch (vertical_alignment) { + case VALIGN_FILL: + case VALIGN_TOP: { + // Nothing. + } break; + case VALIGN_CENTER: { + vbegin = (total_h - line_spacing) / 2.0; + } break; + case VALIGN_BOTTOM: { + vbegin = (total_h - line_spacing); + } break; + } + + WordCache *wc = word_cache; + if (!wc) { + return; + } + + int line = 0; + while (wc) { + if (line >= line_count) { + break; + } + + if (wc->char_pos < 0) { + wc = wc->next; + line++; + continue; + } + + WordCache *from = wc; + WordCache *to = wc; + + float taken = 0; + int spaces = 0; + while (to && to->char_pos >= 0) { + taken += to->pixel_width; + if (to->space_count) { + spaces += to->space_count; + } + to = to->next; + } + + bool can_fill = to && (to->char_pos == WordCache::CHAR_WRAPLINE || to->char_pos == WordCache::CHAR_NEWLINE); + + float x_ofs = 0; + switch (horizontal_alignment) { + case ALIGN_FILL: { + x_ofs = -width / 2.0; + } break; + case ALIGN_LEFT: { + // Noting + } break; + case ALIGN_CENTER: { + x_ofs = -(taken + spaces * space_w) / 2.0; + } break; + case ALIGN_RIGHT: { + x_ofs = -(taken + spaces * space_w); + } break; + } + + float y_ofs = 0; + y_ofs -= line * font_h + font->get_ascent(); + y_ofs += vbegin; + + while (from != to) { + // draw a word + int pos = from->char_pos; + if (from->char_pos < 0) { + ERR_PRINT("BUG"); + return; + } + if (from->space_count) { + /* spacing */ + x_ofs += space_w * from->space_count; + if (can_fill && horizontal_alignment == ALIGN_FILL && spaces) { + x_ofs += ((width - (taken + space_w * spaces)) / spaces); + } + } + + if (font->has_outline()) { + float x_ofs_ol = x_ofs; + for (int i = 0; i < from->word_len; i++) { + CharType c = xl_text[i + pos]; + CharType n = xl_text[i + pos + 1]; + if (uppercase) { + c = String::char_uppercase(c); + n = String::char_uppercase(n); + } + + x_ofs_ol += _generate_glyph_surfaces(font, c, n, lbl_offset + Point2(x_ofs_ol, y_ofs), outline_modulate, outline_render_priority, true); + } + } + for (int i = 0; i < from->word_len; i++) { + CharType c = xl_text[i + pos]; + CharType n = xl_text[i + pos + 1]; + if (uppercase) { + c = String::char_uppercase(c); + n = String::char_uppercase(n); + } + x_ofs += _generate_glyph_surfaces(font, c, n, lbl_offset + Point2(x_ofs, y_ofs), modulate, render_priority, false); + } + from = from->next; + } + + wc = to ? to->next : nullptr; + line++; + } + + const uint64_t *k = nullptr; + int idx = 0; + while ((k = surfaces.next(k))) { + const SurfaceData &surf = surfaces[*k]; + Array mesh_array; + mesh_array.resize(VS::ARRAY_MAX); + mesh_array[VS::ARRAY_VERTEX] = surf.mesh_vertices; + mesh_array[VS::ARRAY_NORMAL] = surf.mesh_normals; + mesh_array[VS::ARRAY_TANGENT] = surf.mesh_tangents; + mesh_array[VS::ARRAY_COLOR] = surf.mesh_colors; + mesh_array[VS::ARRAY_TEX_UV] = surf.mesh_uvs; + mesh_array[VS::ARRAY_INDEX] = surf.indices; + + VS::get_singleton()->mesh_add_surface_from_arrays(mesh, VS::PRIMITIVE_TRIANGLES, mesh_array); + VS::get_singleton()->instance_set_surface_material(get_instance(), idx++, surf.material); + } +} + +void Label3D::set_text(const String &p_string) { + text = p_string; + xl_text = tr(p_string); + word_cache_dirty = true; + _queue_update(); +} + +String Label3D::get_text() const { + return text; +} + +void Label3D::set_horizontal_alignment(Label3D::Align p_alignment) { + ERR_FAIL_INDEX((int)p_alignment, 4); + if (horizontal_alignment != p_alignment) { + horizontal_alignment = p_alignment; + _queue_update(); + } +} + +Label3D::Align Label3D::get_horizontal_alignment() const { + return horizontal_alignment; +} + +void Label3D::set_vertical_alignment(Label3D::VAlign p_alignment) { + ERR_FAIL_INDEX((int)p_alignment, 4); + if (vertical_alignment != p_alignment) { + vertical_alignment = p_alignment; + _queue_update(); + } +} + +Label3D::VAlign Label3D::get_vertical_alignment() const { + return vertical_alignment; +} + +void Label3D::set_uppercase(bool p_uppercase) { + if (uppercase != p_uppercase) { + uppercase = p_uppercase; + word_cache_dirty = true; + _queue_update(); + } +} + +bool Label3D::is_uppercase() const { + return uppercase; +} + +void Label3D::set_render_priority(int p_priority) { + ERR_FAIL_COND(p_priority < VS::MATERIAL_RENDER_PRIORITY_MIN || p_priority > VS::MATERIAL_RENDER_PRIORITY_MAX); + if (render_priority != p_priority) { + render_priority = p_priority; + _queue_update(); + } +} + +int Label3D::get_render_priority() const { + return render_priority; +} + +void Label3D::set_outline_render_priority(int p_priority) { + ERR_FAIL_COND(p_priority < VS::MATERIAL_RENDER_PRIORITY_MIN || p_priority > VS::MATERIAL_RENDER_PRIORITY_MAX); + if (outline_render_priority != p_priority) { + outline_render_priority = p_priority; + _queue_update(); + } +} + +int Label3D::get_outline_render_priority() const { + return outline_render_priority; +} + +void Label3D::_font_changed() { + word_cache_dirty = true; + _queue_update(); +} + +void Label3D::set_font(const Ref &p_font) { + if (font_override != p_font) { + if (font_override.is_valid()) { + font_override->disconnect(CoreStringNames::get_singleton()->changed, this, "_font_changed"); + } + font_override = p_font; + if (font_override.is_valid()) { + font_override->connect(CoreStringNames::get_singleton()->changed, this, "_font_changed"); + } + _queue_update(); + } +} + +Ref Label3D::get_font() const { + return font_override; +} + +Ref Label3D::_get_font_or_default() const { + if (font_override.is_valid()) { + return font_override; + } + + // Check the project-defined Theme resource. + if (Theme::get_project_default().is_valid()) { + List theme_types; + Theme::get_project_default()->get_type_dependencies(get_class_name(), StringName(), &theme_types); + + for (List::Element *E = theme_types.front(); E; E = E->next()) { + if (Theme::get_project_default()->has_theme_item(Theme::DATA_TYPE_FONT, "font", E->get())) { + return Theme::get_project_default()->get_theme_item(Theme::DATA_TYPE_FONT, "font", E->get()); + } + } + } + + // Lastly, fall back on the items defined in the default Theme, if they exist. + { + List theme_types; + Theme::get_default()->get_type_dependencies(get_class_name(), StringName(), &theme_types); + + for (List::Element *E = theme_types.front(); E; E = E->next()) { + if (Theme::get_default()->has_theme_item(Theme::DATA_TYPE_FONT, "font", E->get())) { + return Theme::get_default()->get_theme_item(Theme::DATA_TYPE_FONT, "font", E->get()); + } + } + } + + // If they don't exist, use any type to return the default/empty value. + return Theme::get_default()->get_theme_item(Theme::DATA_TYPE_FONT, "font", StringName()); +} + +void Label3D::set_modulate(const Color &p_color) { + if (modulate != p_color) { + modulate = p_color; + _queue_update(); + } +} + +Color Label3D::get_modulate() const { + return modulate; +} + +void Label3D::set_outline_modulate(const Color &p_color) { + if (outline_modulate != p_color) { + outline_modulate = p_color; + _queue_update(); + } +} + +Color Label3D::get_outline_modulate() const { + return outline_modulate; +} + +void Label3D::set_autowrap(bool p_autowrap) { + if (autowrap != p_autowrap) { + autowrap = p_autowrap; + word_cache_dirty = true; + _queue_update(); + } +} + +bool Label3D::get_autowrap() const { + return autowrap; +} + +void Label3D::set_width(float p_width) { + if (width != p_width) { + width = p_width; + word_cache_dirty = true; + _queue_update(); + } +} + +float Label3D::get_width() const { + return width; +} + +void Label3D::set_pixel_size(real_t p_amount) { + if (pixel_size != p_amount) { + pixel_size = p_amount; + _queue_update(); + } +} + +real_t Label3D::get_pixel_size() const { + return pixel_size; +} + +void Label3D::set_offset(const Point2 &p_offset) { + if (lbl_offset != p_offset) { + lbl_offset = p_offset; + _queue_update(); + } +} + +Point2 Label3D::get_offset() const { + return lbl_offset; +} + +void Label3D::set_line_spacing(float p_line_spacing) { + if (line_spacing != p_line_spacing) { + line_spacing = p_line_spacing; + _queue_update(); + } +} + +float Label3D::get_line_spacing() const { + return line_spacing; +} + +void Label3D::set_draw_flag(DrawFlags p_flag, bool p_enable) { + ERR_FAIL_INDEX(p_flag, FLAG_MAX); + if (flags[p_flag] != p_enable) { + flags[p_flag] = p_enable; + _queue_update(); + } +} + +bool Label3D::get_draw_flag(DrawFlags p_flag) const { + ERR_FAIL_INDEX_V(p_flag, FLAG_MAX, false); + return flags[p_flag]; +} + +void Label3D::set_billboard_mode(SpatialMaterial::BillboardMode p_mode) { + ERR_FAIL_INDEX(p_mode, 3); + if (billboard_mode != p_mode) { + billboard_mode = p_mode; + _queue_update(); + } +} + +SpatialMaterial::BillboardMode Label3D::get_billboard_mode() const { + return billboard_mode; +} + +void Label3D::set_alpha_cut_mode(AlphaCutMode p_mode) { + ERR_FAIL_INDEX(p_mode, 3); + if (alpha_cut != p_mode) { + alpha_cut = p_mode; + _queue_update(); + } +} + +Label3D::AlphaCutMode Label3D::get_alpha_cut_mode() const { + return alpha_cut; +} + +void Label3D::set_alpha_scissor_threshold(float p_threshold) { + if (alpha_scissor_threshold != p_threshold) { + alpha_scissor_threshold = p_threshold; + _queue_update(); + } +} + +float Label3D::get_alpha_scissor_threshold() const { + return alpha_scissor_threshold; +} + +Label3D::Label3D() { + for (int i = 0; i < FLAG_MAX; i++) { + flags[i] = (i == FLAG_DOUBLE_SIDED); + } + + mesh = RID_PRIME(VisualServer::get_singleton()->mesh_create()); + + set_base(mesh); +} + +Label3D::~Label3D() { + while (word_cache) { + WordCache *current = word_cache; + word_cache = current->next; + memdelete(current); + } + + VS::get_singleton()->free(mesh); + const uint64_t *k = nullptr; + while ((k = surfaces.next(k))) { + VS::get_singleton()->free(surfaces[*k].material); + } + surfaces.clear(); +} diff --git a/scene/3d/label_3d.h b/scene/3d/label_3d.h new file mode 100644 index 000000000000..c82e19c622ee --- /dev/null +++ b/scene/3d/label_3d.h @@ -0,0 +1,228 @@ +/*************************************************************************/ +/* label_3d.h */ +/*************************************************************************/ +/* This file is part of: */ +/* GODOT ENGINE */ +/* https://godotengine.org */ +/*************************************************************************/ +/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */ +/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */ +/* */ +/* Permission is hereby granted, free of charge, to any person obtaining */ +/* a copy of this software and associated documentation files (the */ +/* "Software"), to deal in the Software without restriction, including */ +/* without limitation the rights to use, copy, modify, merge, publish, */ +/* distribute, sublicense, and/or sell copies of the Software, and to */ +/* permit persons to whom the Software is furnished to do so, subject to */ +/* the following conditions: */ +/* */ +/* The above copyright notice and this permission notice shall be */ +/* included in all copies or substantial portions of the Software. */ +/* */ +/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ +/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ +/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ +/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ +/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ +/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ +/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ +/*************************************************************************/ + +#ifndef LABEL_3D_H +#define LABEL_3D_H + +#include "scene/3d/visual_instance.h" +#include "scene/resources/font.h" + +class Label3D : public GeometryInstance { + GDCLASS(Label3D, GeometryInstance); + +public: + enum DrawFlags { + FLAG_SHADED, + FLAG_DOUBLE_SIDED, + FLAG_DISABLE_DEPTH_TEST, + FLAG_FIXED_SIZE, + FLAG_MAX + }; + + enum AlphaCutMode { + ALPHA_CUT_DISABLED, + ALPHA_CUT_DISCARD, + ALPHA_CUT_OPAQUE_PREPASS + }; + + enum Align { + + ALIGN_LEFT, + ALIGN_CENTER, + ALIGN_RIGHT, + ALIGN_FILL + }; + + enum VAlign { + + VALIGN_TOP, + VALIGN_CENTER, + VALIGN_BOTTOM, + VALIGN_FILL + }; + +private: + real_t pixel_size = 0.01; + bool flags[FLAG_MAX] = {}; + AlphaCutMode alpha_cut = ALPHA_CUT_DISABLED; + float alpha_scissor_threshold = 0.5; + + AABB aabb; + + mutable Ref triangle_mesh; + RID mesh; + struct SurfaceData { + PoolVector3Array mesh_vertices; + PoolVector3Array mesh_normals; + PoolRealArray mesh_tangents; + PoolColorArray mesh_colors; + PoolVector2Array mesh_uvs; + PoolIntArray indices; + int offset = 0; + float z_shift = 0.0; + RID material; + }; + HashMap surfaces; + + struct WordCache { + enum { + CHAR_NEWLINE = -1, + CHAR_WRAPLINE = -2 + }; + int char_pos; // if -1, then newline + int word_len; + int pixel_width; + int space_count; + WordCache *next; + WordCache() { + char_pos = 0; + word_len = 0; + pixel_width = 0; + next = nullptr; + space_count = 0; + } + }; + bool word_cache_dirty = true; + + WordCache *word_cache = nullptr; + int line_count = 0; + + Align horizontal_alignment = ALIGN_CENTER; + VAlign vertical_alignment = VALIGN_CENTER; + String text; + String xl_text; + bool uppercase = false; + + bool autowrap = false; + float width = 500.0; + + Ref font_override; + Color modulate = Color(1, 1, 1, 1); + Point2 lbl_offset; + int outline_render_priority = -1; + int render_priority = 0; + + Color outline_modulate = Color(0, 0, 0, 1); + + float line_spacing = 0.f; + + RID base_material; + SpatialMaterial::BillboardMode billboard_mode = SpatialMaterial::BILLBOARD_DISABLED; + + bool pending_update = false; + + void regenerate_word_cache(); + int get_longest_line_width() const; + float _generate_glyph_surfaces(const Ref &p_font, CharType p_char, CharType p_next, Vector2 p_offset, const Color &p_modulate, int p_priority, bool p_outline); + +protected: + void _notification(int p_what); + + static void _bind_methods(); + + void _validate_property(PropertyInfo &property) const; + + void _im_update(); + void _font_changed(); + void _queue_update(); + + void _shape(); + +public: + void set_horizontal_alignment(Align p_alignment); + Align get_horizontal_alignment() const; + + void set_vertical_alignment(VAlign p_alignment); + VAlign get_vertical_alignment() const; + + void set_render_priority(int p_priority); + int get_render_priority() const; + + void set_outline_render_priority(int p_priority); + int get_outline_render_priority() const; + + void set_text(const String &p_string); + String get_text() const; + + void set_uppercase(bool p_uppercase); + bool is_uppercase() const; + + void set_font(const Ref &p_font); + Ref get_font() const; + Ref _get_font_or_default() const; + + void set_line_spacing(float p_size); + float get_line_spacing() const; + + void set_modulate(const Color &p_color); + Color get_modulate() const; + + void set_outline_modulate(const Color &p_color); + Color get_outline_modulate() const; + + void set_autowrap(bool p_mode); + bool get_autowrap() const; + + void set_width(float p_width); + float get_width() const; + + void set_pixel_size(real_t p_amount); + real_t get_pixel_size() const; + + void set_offset(const Point2 &p_offset); + Point2 get_offset() const; + + void set_draw_flag(DrawFlags p_flag, bool p_enable); + bool get_draw_flag(DrawFlags p_flag) const; + + void set_alpha_cut_mode(AlphaCutMode p_mode); + AlphaCutMode get_alpha_cut_mode() const; + + void set_alpha_scissor_threshold(float p_threshold); + float get_alpha_scissor_threshold() const; + + void set_billboard_mode(SpatialMaterial::BillboardMode p_mode); + SpatialMaterial::BillboardMode get_billboard_mode() const; + + virtual AABB get_aabb() const; + Ref generate_triangle_mesh() const; + + virtual PoolVector get_faces(uint32_t p_usage_flags) const; + + Label3D(); + ~Label3D(); +}; + +VARIANT_ENUM_CAST(Label3D::DrawFlags); +VARIANT_ENUM_CAST(Label3D::AlphaCutMode); +VARIANT_ENUM_CAST(Label3D::Align); +VARIANT_ENUM_CAST(Label3D::VAlign); + +#endif // LABEL_3D_H diff --git a/scene/3d/sprite_3d.cpp b/scene/3d/sprite_3d.cpp index 9adef13574be..22be5ae4c332 100644 --- a/scene/3d/sprite_3d.cpp +++ b/scene/3d/sprite_3d.cpp @@ -132,6 +132,16 @@ Color SpriteBase3D::get_modulate() const { return modulate; } +void SpriteBase3D::set_render_priority(int p_priority) { + ERR_FAIL_COND(p_priority < VS::MATERIAL_RENDER_PRIORITY_MIN || p_priority > VS::MATERIAL_RENDER_PRIORITY_MAX); + render_priority = p_priority; + _queue_update(); +} + +int SpriteBase3D::get_render_priority() const { + return render_priority; +} + void SpriteBase3D::set_pixel_size(float p_amount) { pixel_size = p_amount; _queue_update(); @@ -296,6 +306,9 @@ void SpriteBase3D::_bind_methods() { ClassDB::bind_method(D_METHOD("set_opacity", "opacity"), &SpriteBase3D::set_opacity); ClassDB::bind_method(D_METHOD("get_opacity"), &SpriteBase3D::get_opacity); + ClassDB::bind_method(D_METHOD("set_render_priority", "priority"), &SpriteBase3D::set_render_priority); + ClassDB::bind_method(D_METHOD("get_render_priority"), &SpriteBase3D::get_render_priority); + ClassDB::bind_method(D_METHOD("set_pixel_size", "pixel_size"), &SpriteBase3D::set_pixel_size); ClassDB::bind_method(D_METHOD("get_pixel_size"), &SpriteBase3D::get_pixel_size); @@ -330,11 +343,16 @@ void SpriteBase3D::_bind_methods() { ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "transparent"), "set_draw_flag", "get_draw_flag", FLAG_TRANSPARENT); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "shaded"), "set_draw_flag", "get_draw_flag", FLAG_SHADED); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "double_sided"), "set_draw_flag", "get_draw_flag", FLAG_DOUBLE_SIDED); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "no_depth_test"), "set_draw_flag", "get_draw_flag", FLAG_DISABLE_DEPTH_TEST); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "fixed_size"), "set_draw_flag", "get_draw_flag", FLAG_FIXED_SIZE); ADD_PROPERTY(PropertyInfo(Variant::INT, "alpha_cut", PROPERTY_HINT_ENUM, "Disabled,Discard,Opaque Pre-Pass"), "set_alpha_cut_mode", "get_alpha_cut_mode"); + ADD_PROPERTY(PropertyInfo(Variant::INT, "render_priority", PROPERTY_HINT_RANGE, itos(VS::MATERIAL_RENDER_PRIORITY_MIN) + "," + itos(VS::MATERIAL_RENDER_PRIORITY_MAX) + ",1"), "set_render_priority", "get_render_priority"); BIND_ENUM_CONSTANT(FLAG_TRANSPARENT); BIND_ENUM_CONSTANT(FLAG_SHADED); BIND_ENUM_CONSTANT(FLAG_DOUBLE_SIDED); + BIND_ENUM_CONSTANT(FLAG_DISABLE_DEPTH_TEST); + BIND_ENUM_CONSTANT(FLAG_FIXED_SIZE); BIND_ENUM_CONSTANT(FLAG_MAX); BIND_ENUM_CONSTANT(ALPHA_CUT_DISABLED); @@ -581,9 +599,12 @@ void Sprite3D::_draw() { VS::get_singleton()->mesh_set_custom_aabb(mesh, aabb); set_aabb(aabb); - RID mat = SpatialMaterial::get_material_rid_for_2d(get_draw_flag(FLAG_SHADED), get_draw_flag(FLAG_TRANSPARENT), get_draw_flag(FLAG_DOUBLE_SIDED), get_alpha_cut_mode() == ALPHA_CUT_DISCARD, get_alpha_cut_mode() == ALPHA_CUT_OPAQUE_PREPASS, get_billboard_mode() == SpatialMaterial::BILLBOARD_ENABLED, get_billboard_mode() == SpatialMaterial::BILLBOARD_FIXED_Y); + RID mat = SpatialMaterial::get_material_rid_for_2d(get_draw_flag(FLAG_SHADED), get_draw_flag(FLAG_TRANSPARENT), get_draw_flag(FLAG_DOUBLE_SIDED), get_alpha_cut_mode() == ALPHA_CUT_DISCARD, get_alpha_cut_mode() == ALPHA_CUT_OPAQUE_PREPASS, get_billboard_mode() == SpatialMaterial::BILLBOARD_ENABLED, get_billboard_mode() == SpatialMaterial::BILLBOARD_FIXED_Y, get_draw_flag(FLAG_DISABLE_DEPTH_TEST), get_draw_flag(FLAG_FIXED_SIZE)); VS::get_singleton()->material_set_shader(get_material(), VS::get_singleton()->material_get_shader(mat)); VS::get_singleton()->material_set_param(get_material(), "texture_albedo", texture->get_rid()); + if (get_alpha_cut_mode() == ALPHA_CUT_DISABLED) { + VS::get_singleton()->material_set_render_priority(get_material(), get_render_priority()); + } VS::get_singleton()->instance_set_surface_material(get_instance(), 0, get_material()); } @@ -924,9 +945,12 @@ void AnimatedSprite3D::_draw() { VS::get_singleton()->mesh_set_custom_aabb(mesh, aabb); set_aabb(aabb); - RID mat = SpatialMaterial::get_material_rid_for_2d(get_draw_flag(FLAG_SHADED), get_draw_flag(FLAG_TRANSPARENT), get_draw_flag(FLAG_DOUBLE_SIDED), get_alpha_cut_mode() == ALPHA_CUT_DISCARD, get_alpha_cut_mode() == ALPHA_CUT_OPAQUE_PREPASS, get_billboard_mode() == SpatialMaterial::BILLBOARD_ENABLED, get_billboard_mode() == SpatialMaterial::BILLBOARD_FIXED_Y); + RID mat = SpatialMaterial::get_material_rid_for_2d(get_draw_flag(FLAG_SHADED), get_draw_flag(FLAG_TRANSPARENT), get_draw_flag(FLAG_DOUBLE_SIDED), get_alpha_cut_mode() == ALPHA_CUT_DISCARD, get_alpha_cut_mode() == ALPHA_CUT_OPAQUE_PREPASS, get_billboard_mode() == SpatialMaterial::BILLBOARD_ENABLED, get_billboard_mode() == SpatialMaterial::BILLBOARD_FIXED_Y, get_draw_flag(FLAG_DISABLE_DEPTH_TEST), get_draw_flag(FLAG_FIXED_SIZE)); VS::get_singleton()->material_set_shader(get_material(), VS::get_singleton()->material_get_shader(mat)); VS::get_singleton()->material_set_param(get_material(), "texture_albedo", texture->get_rid()); + if (get_alpha_cut_mode() == ALPHA_CUT_DISABLED) { + VS::get_singleton()->material_set_render_priority(get_material(), get_render_priority()); + } VS::get_singleton()->instance_set_surface_material(get_instance(), 0, get_material()); } diff --git a/scene/3d/sprite_3d.h b/scene/3d/sprite_3d.h index 9e7230d706e7..163ff2f10e52 100644 --- a/scene/3d/sprite_3d.h +++ b/scene/3d/sprite_3d.h @@ -44,6 +44,8 @@ class SpriteBase3D : public GeometryInstance { FLAG_TRANSPARENT, FLAG_SHADED, FLAG_DOUBLE_SIDED, + FLAG_DISABLE_DEPTH_TEST, + FLAG_FIXED_SIZE, FLAG_MAX }; @@ -69,6 +71,7 @@ class SpriteBase3D : public GeometryInstance { bool vflip; Color modulate; + int render_priority = 0; float opacity; Vector3::Axis axis; @@ -121,6 +124,9 @@ class SpriteBase3D : public GeometryInstance { void set_opacity(float p_amount); float get_opacity() const; + void set_render_priority(int p_priority); + int get_render_priority() const; + void set_pixel_size(float p_amount); float get_pixel_size() const; diff --git a/scene/gui/label.cpp b/scene/gui/label.cpp index 677803e19ae0..9dcafcc51299 100644 --- a/scene/gui/label.cpp +++ b/scene/gui/label.cpp @@ -195,7 +195,7 @@ void Label::_notification(int p_what) { to = to->next; } - bool can_fill = to && to->char_pos == WordCache::CHAR_WRAPLINE; + bool can_fill = to && (to->char_pos == WordCache::CHAR_WRAPLINE || to->char_pos == WordCache::CHAR_NEWLINE); float x_ofs = 0; diff --git a/scene/register_scene_types.cpp b/scene/register_scene_types.cpp index fe5626b14e16..8ee741939db6 100644 --- a/scene/register_scene_types.cpp +++ b/scene/register_scene_types.cpp @@ -190,6 +190,7 @@ #include "scene/3d/gi_probe.h" #include "scene/3d/immediate_geometry.h" #include "scene/3d/interpolated_camera.h" +#include "scene/3d/label_3d.h" #include "scene/3d/light.h" #include "scene/3d/listener.h" #include "scene/3d/mesh_instance.h" @@ -441,6 +442,7 @@ void register_scene_types() { ClassDB::register_virtual_class(); ClassDB::register_class(); ClassDB::register_class(); + ClassDB::register_class(); ClassDB::register_virtual_class(); ClassDB::register_class(); ClassDB::register_class(); diff --git a/scene/resources/dynamic_font.cpp b/scene/resources/dynamic_font.cpp index 0a242d4b30cc..826c31e531ca 100644 --- a/scene/resources/dynamic_font.cpp +++ b/scene/resources/dynamic_font.cpp @@ -334,6 +334,147 @@ void DynamicFontAtSize::set_texture_flags(uint32_t p_flags) { } } +RID DynamicFontAtSize::get_char_texture(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const { + if (!valid) { + return RID(); + } + + int32_t c = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + c = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return RID(); + } + + const_cast(this)->_update_char(c); + + Pair char_pair_with_font = _find_char_with_font(c, p_fallbacks); + const Character *ch = char_pair_with_font.first; + DynamicFontAtSize *font = char_pair_with_font.second; + + ERR_FAIL_COND_V(!ch, RID()); + if (ch->found) { + ERR_FAIL_COND_V(ch->texture_idx < -1 || ch->texture_idx >= font->textures.size(), RID()); + + if (ch->texture_idx != -1) { + return font->textures[ch->texture_idx].texture->get_rid(); + } + } + return RID(); +} + +Size2 DynamicFontAtSize::get_char_texture_size(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const { + if (!valid) { + return Size2(); + } + + int32_t c = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + c = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Size2(); + } + + const_cast(this)->_update_char(c); + + Pair char_pair_with_font = _find_char_with_font(c, p_fallbacks); + const Character *ch = char_pair_with_font.first; + DynamicFontAtSize *font = char_pair_with_font.second; + + ERR_FAIL_COND_V(!ch, Size2()); + if (ch->found) { + ERR_FAIL_COND_V(ch->texture_idx < -1 || ch->texture_idx >= font->textures.size(), Size2()); + + if (ch->texture_idx != -1) { + return font->textures[ch->texture_idx].texture->get_size(); + } + } + return Size2(); +} + +Vector2 DynamicFontAtSize::get_char_tx_offset(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const { + if (!valid) { + return Vector2(); + } + + int32_t c = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + c = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Vector2(); + } + + const_cast(this)->_update_char(c); + + Pair char_pair_with_font = _find_char_with_font(c, p_fallbacks); + const Character *ch = char_pair_with_font.first; + DynamicFontAtSize *font = char_pair_with_font.second; + + ERR_FAIL_COND_V(!ch, Vector2()); + if (ch->found) { + Point2 cpos; + cpos.x += ch->h_align; + cpos.y -= font->get_ascent(); + cpos.y += ch->v_align; + + return cpos; + } + return Vector2(); +} + +Size2 DynamicFontAtSize::get_char_tx_size(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const { + if (!valid) { + return Size2(); + } + + int32_t c = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + c = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Size2(); + } + + const_cast(this)->_update_char(c); + + Pair char_pair_with_font = _find_char_with_font(c, p_fallbacks); + const Character *ch = char_pair_with_font.first; + + ERR_FAIL_COND_V(!ch, Size2()); + if (ch->found) { + return ch->rect_uv.size; + } + return Size2(); +} + +Rect2 DynamicFontAtSize::get_char_tx_uv_rect(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const { + if (!valid) { + return Rect2(); + } + + int32_t c = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + c = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Rect2(); + } + + const_cast(this)->_update_char(c); + + Pair char_pair_with_font = _find_char_with_font(c, p_fallbacks); + const Character *ch = char_pair_with_font.first; + + ERR_FAIL_COND_V(!ch, Rect2()); + if (ch->found) { + return ch->rect_uv; + } + return Rect2(); +} + float DynamicFontAtSize::draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next, const Color &p_modulate, const Vector> &p_fallbacks, bool p_advance_only, bool p_outline) const { if (!valid) { return 0; @@ -942,6 +1083,81 @@ bool DynamicFont::has_outline() const { return outline_cache_id.outline_size > 0; } +RID DynamicFont::get_char_texture(CharType p_char, CharType p_next, bool p_outline) const { + if (!data_at_size.is_valid()) { + return RID(); + } + + if (p_outline) { + if (outline_data_at_size.is_valid() && outline_cache_id.outline_size > 0) { + return outline_data_at_size->get_char_texture(p_char, p_next, fallback_outline_data_at_size); + } + return RID(); + } else { + return data_at_size->get_char_texture(p_char, p_next, fallback_data_at_size); + } +} + +Size2 DynamicFont::get_char_texture_size(CharType p_char, CharType p_next, bool p_outline) const { + if (!data_at_size.is_valid()) { + return Size2(); + } + + if (p_outline) { + if (outline_data_at_size.is_valid() && outline_cache_id.outline_size > 0) { + return outline_data_at_size->get_char_texture_size(p_char, p_next, fallback_outline_data_at_size); + } + return Size2(); + } else { + return data_at_size->get_char_texture_size(p_char, p_next, fallback_data_at_size); + } +} + +Vector2 DynamicFont::get_char_tx_offset(CharType p_char, CharType p_next, bool p_outline) const { + if (!data_at_size.is_valid()) { + return Vector2(); + } + + if (p_outline) { + if (outline_data_at_size.is_valid() && outline_cache_id.outline_size > 0) { + return outline_data_at_size->get_char_tx_offset(p_char, p_next, fallback_outline_data_at_size); + } + return Vector2(); + } else { + return data_at_size->get_char_tx_offset(p_char, p_next, fallback_data_at_size); + } +} + +Size2 DynamicFont::get_char_tx_size(CharType p_char, CharType p_next, bool p_outline) const { + if (!data_at_size.is_valid()) { + return Size2(); + } + + if (p_outline) { + if (outline_data_at_size.is_valid() && outline_cache_id.outline_size > 0) { + return outline_data_at_size->get_char_tx_size(p_char, p_next, fallback_outline_data_at_size); + } + return Size2(); + } else { + return data_at_size->get_char_tx_size(p_char, p_next, fallback_data_at_size); + } +} + +Rect2 DynamicFont::get_char_tx_uv_rect(CharType p_char, CharType p_next, bool p_outline) const { + if (!data_at_size.is_valid()) { + return Rect2(); + } + + if (p_outline) { + if (outline_data_at_size.is_valid() && outline_cache_id.outline_size > 0) { + return outline_data_at_size->get_char_tx_uv_rect(p_char, p_next, fallback_outline_data_at_size); + } + return Rect2(); + } else { + return data_at_size->get_char_tx_uv_rect(p_char, p_next, fallback_data_at_size); + } +} + float DynamicFont::draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next, const Color &p_modulate, bool p_outline) const { if (!data_at_size.is_valid()) { return 0; diff --git a/scene/resources/dynamic_font.h b/scene/resources/dynamic_font.h index 111f2b4c1f89..eaea7dc86f7a 100644 --- a/scene/resources/dynamic_font.h +++ b/scene/resources/dynamic_font.h @@ -195,6 +195,13 @@ class DynamicFontAtSize : public Reference { float draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next, const Color &p_modulate, const Vector> &p_fallbacks, bool p_advance_only = false, bool p_outline = false) const; + RID get_char_texture(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const; + Size2 get_char_texture_size(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const; + + Vector2 get_char_tx_offset(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const; + Size2 get_char_tx_size(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const; + Rect2 get_char_tx_uv_rect(CharType p_char, CharType p_next, const Vector> &p_fallbacks) const; + void set_texture_flags(uint32_t p_flags); void update_oversampling(); @@ -286,6 +293,13 @@ class DynamicFont : public Font { virtual float draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next = 0, const Color &p_modulate = Color(1, 1, 1), bool p_outline = false) const; + RID get_char_texture(CharType p_char, CharType p_next, bool p_outline) const; + Size2 get_char_texture_size(CharType p_char, CharType p_next, bool p_outline) const; + + Vector2 get_char_tx_offset(CharType p_char, CharType p_next, bool p_outline) const; + Size2 get_char_tx_size(CharType p_char, CharType p_next, bool p_outline) const; + Rect2 get_char_tx_uv_rect(CharType p_char, CharType p_next, bool p_outline) const; + SelfList font_list; static Mutex dynamic_font_mutex; diff --git a/scene/resources/font.cpp b/scene/resources/font.cpp index 130ffda150ea..d479e92f3620 100644 --- a/scene/resources/font.cpp +++ b/scene/resources/font.cpp @@ -98,6 +98,13 @@ void Font::_bind_methods() { ClassDB::bind_method(D_METHOD("get_wordwrap_string_size", "string", "width"), &Font::get_wordwrap_string_size); ClassDB::bind_method(D_METHOD("has_outline"), &Font::has_outline); ClassDB::bind_method(D_METHOD("draw_char", "canvas_item", "position", "char", "next", "modulate", "outline"), &Font::draw_char, DEFVAL(-1), DEFVAL(Color(1, 1, 1)), DEFVAL(false)); + + ClassDB::bind_method(D_METHOD("get_char_texture", "char", "next", "outline"), &Font::get_char_texture, DEFVAL(0), DEFVAL(false)); + ClassDB::bind_method(D_METHOD("get_char_texture_size", "char", "next", "outline"), &Font::get_char_texture_size, DEFVAL(0), DEFVAL(false)); + ClassDB::bind_method(D_METHOD("get_char_tx_offset", "char", "next", "outline"), &Font::get_char_tx_offset, DEFVAL(0), DEFVAL(false)); + ClassDB::bind_method(D_METHOD("get_char_tx_size", "char", "next", "outline"), &Font::get_char_tx_size, DEFVAL(0), DEFVAL(false)); + ClassDB::bind_method(D_METHOD("get_char_tx_uv_rect", "char", "next", "outline"), &Font::get_char_tx_uv_rect, DEFVAL(0), DEFVAL(false)); + ClassDB::bind_method(D_METHOD("update_changes"), &Font::update_changes); } @@ -523,6 +530,140 @@ Ref BitmapFont::get_fallback() const { return fallback; } +RID BitmapFont::get_char_texture(CharType p_char, CharType p_next, bool p_outline) const { + int32_t ch = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + ch = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return RID(); + } + + const Character *c = char_map.getptr(ch); + + if (!c) { + if (fallback.is_valid()) { + return fallback->get_char_texture(p_char, p_next, p_outline); + } + return RID(); + } + + ERR_FAIL_COND_V(c->texture_idx < -1 || c->texture_idx >= textures.size(), RID()); + if (!p_outline && c->texture_idx != -1) { + return textures[c->texture_idx]->get_rid(); + } else { + return RID(); + } +} + +Size2 BitmapFont::get_char_texture_size(CharType p_char, CharType p_next, bool p_outline) const { + int32_t ch = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + ch = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Size2(); + } + + const Character *c = char_map.getptr(ch); + + if (!c) { + if (fallback.is_valid()) { + return fallback->get_char_texture_size(p_char, p_next, p_outline); + } + return Size2(); + } + + ERR_FAIL_COND_V(c->texture_idx < -1 || c->texture_idx >= textures.size(), Size2()); + if (!p_outline && c->texture_idx != -1) { + return textures[c->texture_idx]->get_size(); + } else { + return Size2(); + } +} + +Vector2 BitmapFont::get_char_tx_offset(CharType p_char, CharType p_next, bool p_outline) const { + int32_t ch = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + ch = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Vector2(); + } + + const Character *c = char_map.getptr(ch); + + if (!c) { + if (fallback.is_valid()) { + return fallback->get_char_tx_offset(p_char, p_next, p_outline); + } + return Vector2(); + } + + ERR_FAIL_COND_V(c->texture_idx < -1 || c->texture_idx >= textures.size(), Vector2()); + if (!p_outline && c->texture_idx != -1) { + Point2 cpos; + cpos.x += c->h_align; + cpos.y -= ascent; + cpos.y += c->v_align; + return cpos; + } else { + return Vector2(); + } +} + +Size2 BitmapFont::get_char_tx_size(CharType p_char, CharType p_next, bool p_outline) const { + int32_t ch = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + ch = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Size2(); + } + + const Character *c = char_map.getptr(ch); + + if (!c) { + if (fallback.is_valid()) { + return fallback->get_char_tx_size(p_char, p_next, p_outline); + } + return Size2(); + } + + ERR_FAIL_COND_V(c->texture_idx < -1 || c->texture_idx >= textures.size(), Size2()); + if (!p_outline && c->texture_idx != -1) { + return c->rect.size; + } else { + return Size2(); + } +} + +Rect2 BitmapFont::get_char_tx_uv_rect(CharType p_char, CharType p_next, bool p_outline) const { + int32_t ch = p_char; + if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. + ch = (p_char << 10UL) + p_next - ((0xd800 << 10UL) + 0xdc00 - 0x10000); + } + if ((p_char & 0xfffffc00) == 0xdc00) { // skip trail surrogate. + return Rect2(); + } + + const Character *c = char_map.getptr(ch); + + if (!c) { + if (fallback.is_valid()) { + return fallback->get_char_tx_uv_rect(p_char, p_next, p_outline); + } + return Rect2(); + } + + ERR_FAIL_COND_V(c->texture_idx < -1 || c->texture_idx >= textures.size(), Rect2()); + if (!p_outline && c->texture_idx != -1) { + return c->rect; + } else { + return Rect2(); + } +} + float BitmapFont::draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next, const Color &p_modulate, bool p_outline) const { int32_t ch = p_char; if (((p_char & 0xfffffc00) == 0xd800) && (p_next & 0xfffffc00) == 0xdc00) { // decode surrogate pair. diff --git a/scene/resources/font.h b/scene/resources/font.h index fd6db9799679..947af7b50574 100644 --- a/scene/resources/font.h +++ b/scene/resources/font.h @@ -59,6 +59,13 @@ class Font : public Resource { virtual bool has_outline() const { return false; } virtual float draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next = 0, const Color &p_modulate = Color(1, 1, 1), bool p_outline = false) const = 0; + virtual RID get_char_texture(CharType p_char, CharType p_next, bool p_outline) const = 0; + virtual Size2 get_char_texture_size(CharType p_char, CharType p_next, bool p_outline) const = 0; + + virtual Vector2 get_char_tx_offset(CharType p_char, CharType p_next, bool p_outline) const = 0; + virtual Size2 get_char_tx_size(CharType p_char, CharType p_next, bool p_outline) const = 0; + virtual Rect2 get_char_tx_uv_rect(CharType p_char, CharType p_next, bool p_outline) const = 0; + void update_changes(); Font(); }; @@ -190,6 +197,13 @@ class BitmapFont : public Font { float draw_char(RID p_canvas_item, const Point2 &p_pos, CharType p_char, CharType p_next = 0, const Color &p_modulate = Color(1, 1, 1), bool p_outline = false) const; + RID get_char_texture(CharType p_char, CharType p_next, bool p_outline) const; + Size2 get_char_texture_size(CharType p_char, CharType p_next, bool p_outline) const; + + Vector2 get_char_tx_offset(CharType p_char, CharType p_next, bool p_outline) const; + Size2 get_char_tx_size(CharType p_char, CharType p_next, bool p_outline) const; + Rect2 get_char_tx_uv_rect(CharType p_char, CharType p_next, bool p_outline) const; + BitmapFont(); ~BitmapFont(); }; diff --git a/scene/resources/material.cpp b/scene/resources/material.cpp index 6cee213d2379..82697e43cdf7 100644 --- a/scene/resources/material.cpp +++ b/scene/resources/material.cpp @@ -347,12 +347,10 @@ void SpatialMaterial::init_shaders() { shader_names->texture_names[TEXTURE_DETAIL_NORMAL] = "texture_detail_normal"; } -Ref SpatialMaterial::materials_for_2d[SpatialMaterial::MAX_MATERIALS_FOR_2D]; +HashMap> SpatialMaterial::materials_for_2d; void SpatialMaterial::finish_shaders() { - for (int i = 0; i < MAX_MATERIALS_FOR_2D; i++) { - materials_for_2d[i].unref(); - } + materials_for_2d.clear(); memdelete(dirty_materials); dirty_materials = nullptr; @@ -813,7 +811,12 @@ void SpatialMaterial::_update_shader() { } } - if (flags[FLAG_ALBEDO_TEXTURE_FORCE_SRGB]) { + if (flags[FLAG_ALBEDO_TEXTURE_SDF]) { + code += "\tconst float smoothing = 0.125;\n"; + code += "\tfloat dist = albedo_tex.a;\n"; + code += "\talbedo_tex.a = smoothstep(0.5 - smoothing, 0.5 + smoothing, dist);\n"; + code += "\talbedo_tex.rgb = vec3(1.0);\n"; + } else if (flags[FLAG_ALBEDO_TEXTURE_FORCE_SRGB]) { code += "\talbedo_tex.rgb = mix(pow((albedo_tex.rgb + vec3(0.055)) * (1.0 / (1.0 + 0.055)),vec3(2.4)),albedo_tex.rgb.rgb * (1.0 / 12.92),lessThan(albedo_tex.rgb,vec3(0.04045)));\n"; } @@ -1728,32 +1731,41 @@ SpatialMaterial::TextureChannel SpatialMaterial::get_refraction_texture_channel( return refraction_texture_channel; } -RID SpatialMaterial::get_material_rid_for_2d(bool p_shaded, bool p_transparent, bool p_double_sided, bool p_cut_alpha, bool p_opaque_prepass, bool p_billboard, bool p_billboard_y) { - int version = 0; +RID SpatialMaterial::get_material_rid_for_2d(bool p_shaded, bool p_transparent, bool p_double_sided, bool p_cut_alpha, bool p_opaque_prepass, bool p_billboard, bool p_billboard_y, bool p_no_depth_test, bool p_fixed_size, bool p_sdf) { + uint64_t hash = 0; if (p_shaded) { - version = 1; + hash |= 1 << 0; } if (p_transparent) { - version |= 2; + hash |= 1 << 1; } if (p_cut_alpha) { - version |= 4; + hash |= 1 << 2; } if (p_opaque_prepass) { - version |= 8; + hash |= 1 << 3; } if (p_double_sided) { - version |= 16; + hash |= 1 << 4; } if (p_billboard) { - version |= 32; + hash |= 1 << 5; } if (p_billboard_y) { - version |= 64; + hash |= 1 << 6; + } + if (p_no_depth_test) { + hash |= 1 << 7; + } + if (p_fixed_size) { + hash |= 1 << 8; + } + if (p_sdf) { + hash |= 1 << 9; } - if (materials_for_2d[version].is_valid()) { - return materials_for_2d[version]->get_rid(); + if (materials_for_2d.has(hash)) { + return materials_for_2d[hash]->get_rid(); } Ref material; @@ -1766,16 +1778,19 @@ RID SpatialMaterial::get_material_rid_for_2d(bool p_shaded, bool p_transparent, material->set_flag(FLAG_SRGB_VERTEX_COLOR, true); material->set_flag(FLAG_ALBEDO_FROM_VERTEX_COLOR, true); material->set_flag(FLAG_USE_ALPHA_SCISSOR, p_cut_alpha); + material->set_flag(FLAG_DISABLE_DEPTH_TEST, p_no_depth_test); + material->set_flag(FLAG_FIXED_SIZE, p_fixed_size); + material->set_flag(FLAG_ALBEDO_TEXTURE_SDF, p_sdf); if (p_billboard || p_billboard_y) { material->set_flag(FLAG_BILLBOARD_KEEP_SCALE, true); material->set_billboard_mode(p_billboard_y ? BILLBOARD_FIXED_Y : BILLBOARD_ENABLED); } - materials_for_2d[version] = material; + materials_for_2d[hash] = material; // flush before using so we can access the shader right away flush_changes(); - return materials_for_2d[version]->get_rid(); + return materials_for_2d[hash]->get_rid(); } void SpatialMaterial::set_on_top_of_alpha() { @@ -2047,6 +2062,7 @@ void SpatialMaterial::_bind_methods() { ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "flags_do_not_receive_shadows"), "set_flag", "get_flag", FLAG_DONT_RECEIVE_SHADOWS); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "flags_disable_ambient_light"), "set_flag", "get_flag", FLAG_DISABLE_AMBIENT_LIGHT); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "flags_ensure_correct_normals"), "set_flag", "get_flag", FLAG_ENSURE_CORRECT_NORMALS); + ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "flags_albedo_tex_msdf"), "set_flag", "get_flag", FLAG_ALBEDO_TEXTURE_SDF); ADD_GROUP("Vertex Color", "vertex_color"); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "vertex_color_use_as_albedo"), "set_flag", "get_flag", FLAG_ALBEDO_FROM_VERTEX_COLOR); ADD_PROPERTYI(PropertyInfo(Variant::BOOL, "vertex_color_is_srgb"), "set_flag", "get_flag", FLAG_SRGB_VERTEX_COLOR); @@ -2246,6 +2262,7 @@ void SpatialMaterial::_bind_methods() { BIND_ENUM_CONSTANT(FLAG_DISABLE_AMBIENT_LIGHT); BIND_ENUM_CONSTANT(FLAG_ENSURE_CORRECT_NORMALS); BIND_ENUM_CONSTANT(FLAG_USE_SHADOW_TO_OPACITY); + BIND_ENUM_CONSTANT(FLAG_ALBEDO_TEXTURE_SDF); BIND_ENUM_CONSTANT(FLAG_MAX); BIND_ENUM_CONSTANT(DIFFUSE_BURLEY); diff --git a/scene/resources/material.h b/scene/resources/material.h index aadede205a3d..3d9f5b04d2c3 100644 --- a/scene/resources/material.h +++ b/scene/resources/material.h @@ -191,6 +191,7 @@ class SpatialMaterial : public Material { FLAG_ENSURE_CORRECT_NORMALS, FLAG_DISABLE_AMBIENT_LIGHT, FLAG_USE_SHADOW_TO_OPACITY, + FLAG_ALBEDO_TEXTURE_SDF, FLAG_MAX }; @@ -445,9 +446,7 @@ class SpatialMaterial : public Material { _FORCE_INLINE_ void _validate_feature(const String &text, Feature feature, PropertyInfo &property) const; - static const int MAX_MATERIALS_FOR_2D = 128; - - static Ref materials_for_2d[MAX_MATERIALS_FOR_2D]; //used by Sprite3D and other stuff + static HashMap> materials_for_2d; //used by Sprite3D and other stuff void _validate_high_end(const String &text, PropertyInfo &property) const; @@ -635,7 +634,7 @@ class SpatialMaterial : public Material { static void finish_shaders(); static void flush_changes(); - static RID get_material_rid_for_2d(bool p_shaded, bool p_transparent, bool p_double_sided, bool p_cut_alpha, bool p_opaque_prepass, bool p_billboard = false, bool p_billboard_y = false); + static RID get_material_rid_for_2d(bool p_shaded, bool p_transparent, bool p_double_sided, bool p_cut_alpha, bool p_opaque_prepass, bool p_billboard = false, bool p_billboard_y = false, bool p_no_depth_test = false, bool p_fixed_size = false, bool p_sdf = false); RID get_shader_rid() const;