mirror of
https://github.com/godotengine/godot.git
synced 2025-01-23 19:12:24 -05:00
Merge pull request #73343 from RandomShaper/labels_are_evil
Make label sizing algorithm more robust
This commit is contained in:
commit
2874961309
2 changed files with 137 additions and 130 deletions
|
@ -86,9 +86,7 @@ int Label::get_line_height(int p_line) const {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Label::_shape() {
|
bool Label::_shape() {
|
||||||
bool font_was_dirty = font_dirty;
|
|
||||||
|
|
||||||
Ref<StyleBox> style = theme_cache.normal_style;
|
Ref<StyleBox> style = theme_cache.normal_style;
|
||||||
int width = (get_size().width - style->get_minimum_size().width);
|
int width = (get_size().width - style->get_minimum_size().width);
|
||||||
|
|
||||||
|
@ -103,7 +101,7 @@ void Label::_shape() {
|
||||||
}
|
}
|
||||||
const Ref<Font> &font = (settings.is_valid() && settings->get_font().is_valid()) ? settings->get_font() : theme_cache.font;
|
const Ref<Font> &font = (settings.is_valid() && settings->get_font().is_valid()) ? settings->get_font() : theme_cache.font;
|
||||||
int font_size = settings.is_valid() ? settings->get_font_size() : theme_cache.font_size;
|
int font_size = settings.is_valid() ? settings->get_font_size() : theme_cache.font_size;
|
||||||
ERR_FAIL_COND(font.is_null());
|
ERR_FAIL_COND_V(font.is_null(), true);
|
||||||
String txt = (uppercase) ? TS->string_to_upper(xl_text, language) : xl_text;
|
String txt = (uppercase) ? TS->string_to_upper(xl_text, language) : xl_text;
|
||||||
if (visible_chars >= 0 && visible_chars_behavior == TextServer::VC_CHARS_BEFORE_SHAPING) {
|
if (visible_chars >= 0 && visible_chars_behavior == TextServer::VC_CHARS_BEFORE_SHAPING) {
|
||||||
txt = txt.substr(0, visible_chars);
|
txt = txt.substr(0, visible_chars);
|
||||||
|
@ -123,6 +121,7 @@ void Label::_shape() {
|
||||||
dirty = false;
|
dirty = false;
|
||||||
font_dirty = false;
|
font_dirty = false;
|
||||||
lines_dirty = true;
|
lines_dirty = true;
|
||||||
|
// Note for future maintainers: forgetting stable width here (e.g., setting it to -1) may fix still undiscovered bugs.
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lines_dirty) {
|
if (lines_dirty) {
|
||||||
|
@ -135,129 +134,136 @@ void Label::_shape() {
|
||||||
Size2i prev_minsize = minsize;
|
Size2i prev_minsize = minsize;
|
||||||
minsize = Size2();
|
minsize = Size2();
|
||||||
|
|
||||||
|
bool can_process_lines = false;
|
||||||
if (xl_text.length() == 0) {
|
if (xl_text.length() == 0) {
|
||||||
|
can_process_lines = true;
|
||||||
lines_dirty = false;
|
lines_dirty = false;
|
||||||
return;
|
} else {
|
||||||
}
|
// With autowrap on, we won't compute the minimum size until width is stable (two shape requests in a
|
||||||
|
// row with the same width.) This avoids situations in which the initial width is very narrow and the label
|
||||||
|
// would break text into many very short lines, causing a very tall label that can leave a deformed container.
|
||||||
|
|
||||||
// Don't compute minimum size until width is stable (two shape requests in a row with the same width.)
|
can_process_lines = get_size().width == stable_width || autowrap_mode == TextServer::AUTOWRAP_OFF;
|
||||||
// This avoids situations in which the initial width is very narrow and the label would break text
|
stable_width = get_size().width;
|
||||||
// into many very short lines, causing a very tall label that can leave a deformed container.
|
|
||||||
bool width_stabilized = get_size().width == stable_width;
|
|
||||||
stable_width = get_size().width;
|
|
||||||
|
|
||||||
// With autowrap off, there's still a point in shaping before width is stable: to contribute a min width.
|
if (can_process_lines) {
|
||||||
if (lines_dirty && (width_stabilized || autowrap_mode == TextServer::AUTOWRAP_OFF)) {
|
if (lines_dirty) {
|
||||||
BitField<TextServer::LineBreakFlag> autowrap_flags = TextServer::BREAK_MANDATORY;
|
BitField<TextServer::LineBreakFlag> autowrap_flags = TextServer::BREAK_MANDATORY;
|
||||||
switch (autowrap_mode) {
|
switch (autowrap_mode) {
|
||||||
case TextServer::AUTOWRAP_WORD_SMART:
|
case TextServer::AUTOWRAP_WORD_SMART:
|
||||||
autowrap_flags = TextServer::BREAK_WORD_BOUND | TextServer::BREAK_ADAPTIVE | TextServer::BREAK_MANDATORY;
|
autowrap_flags = TextServer::BREAK_WORD_BOUND | TextServer::BREAK_ADAPTIVE | TextServer::BREAK_MANDATORY;
|
||||||
break;
|
break;
|
||||||
case TextServer::AUTOWRAP_WORD:
|
case TextServer::AUTOWRAP_WORD:
|
||||||
autowrap_flags = TextServer::BREAK_WORD_BOUND | TextServer::BREAK_MANDATORY;
|
autowrap_flags = TextServer::BREAK_WORD_BOUND | TextServer::BREAK_MANDATORY;
|
||||||
break;
|
break;
|
||||||
case TextServer::AUTOWRAP_ARBITRARY:
|
case TextServer::AUTOWRAP_ARBITRARY:
|
||||||
autowrap_flags = TextServer::BREAK_GRAPHEME_BOUND | TextServer::BREAK_MANDATORY;
|
autowrap_flags = TextServer::BREAK_GRAPHEME_BOUND | TextServer::BREAK_MANDATORY;
|
||||||
break;
|
break;
|
||||||
case TextServer::AUTOWRAP_OFF:
|
case TextServer::AUTOWRAP_OFF:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
autowrap_flags = autowrap_flags | TextServer::BREAK_TRIM_EDGE_SPACES;
|
autowrap_flags = autowrap_flags | TextServer::BREAK_TRIM_EDGE_SPACES;
|
||||||
|
|
||||||
PackedInt32Array line_breaks = TS->shaped_text_get_line_breaks(text_rid, width, 0, autowrap_flags);
|
PackedInt32Array line_breaks = TS->shaped_text_get_line_breaks(text_rid, width, 0, autowrap_flags);
|
||||||
for (int i = 0; i < line_breaks.size(); i = i + 2) {
|
for (int i = 0; i < line_breaks.size(); i = i + 2) {
|
||||||
RID line = TS->shaped_text_substr(text_rid, line_breaks[i], line_breaks[i + 1] - line_breaks[i]);
|
RID line = TS->shaped_text_substr(text_rid, line_breaks[i], line_breaks[i + 1] - line_breaks[i]);
|
||||||
lines_rid.push_back(line);
|
lines_rid.push_back(line);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (autowrap_mode == TextServer::AUTOWRAP_OFF) {
|
|
||||||
for (int i = 0; i < lines_rid.size(); i++) {
|
|
||||||
if (minsize.width < TS->shaped_text_get_size(lines_rid[i]).x) {
|
|
||||||
minsize.width = TS->shaped_text_get_size(lines_rid[i]).x;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// With autowrap off, by now we already know the width the label will take.
|
if (autowrap_mode == TextServer::AUTOWRAP_OFF) {
|
||||||
width = (minsize.width - style->get_minimum_size().width);
|
|
||||||
width_stabilized = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (lines_dirty && width_stabilized) {
|
|
||||||
BitField<TextServer::TextOverrunFlag> overrun_flags = TextServer::OVERRUN_NO_TRIM;
|
|
||||||
switch (overrun_behavior) {
|
|
||||||
case TextServer::OVERRUN_TRIM_WORD_ELLIPSIS:
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM_WORD_ONLY);
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_ADD_ELLIPSIS);
|
|
||||||
break;
|
|
||||||
case TextServer::OVERRUN_TRIM_ELLIPSIS:
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_ADD_ELLIPSIS);
|
|
||||||
break;
|
|
||||||
case TextServer::OVERRUN_TRIM_WORD:
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM_WORD_ONLY);
|
|
||||||
break;
|
|
||||||
case TextServer::OVERRUN_TRIM_CHAR:
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
|
||||||
break;
|
|
||||||
case TextServer::OVERRUN_NO_TRIMMING:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fill after min_size calculation.
|
|
||||||
|
|
||||||
int visible_lines = lines_rid.size();
|
|
||||||
if (max_lines_visible >= 0 && visible_lines > max_lines_visible) {
|
|
||||||
visible_lines = max_lines_visible;
|
|
||||||
}
|
|
||||||
if (autowrap_mode != TextServer::AUTOWRAP_OFF) {
|
|
||||||
bool lines_hidden = visible_lines > 0 && visible_lines < lines_rid.size();
|
|
||||||
if (lines_hidden) {
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_ENFORCE_ELLIPSIS);
|
|
||||||
}
|
|
||||||
if (horizontal_alignment == HORIZONTAL_ALIGNMENT_FILL) {
|
|
||||||
for (int i = 0; i < lines_rid.size(); i++) {
|
for (int i = 0; i < lines_rid.size(); i++) {
|
||||||
if (i < visible_lines - 1 || lines_rid.size() == 1) {
|
if (minsize.width < TS->shaped_text_get_size(lines_rid[i]).x) {
|
||||||
TS->shaped_text_fit_to_width(lines_rid[i], width);
|
minsize.width = TS->shaped_text_get_size(lines_rid[i]).x;
|
||||||
} else if (i == (visible_lines - 1)) {
|
|
||||||
TS->shaped_text_overrun_trim_to_width(lines_rid[visible_lines - 1], width, overrun_flags);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (lines_hidden) {
|
|
||||||
TS->shaped_text_overrun_trim_to_width(lines_rid[visible_lines - 1], width, overrun_flags);
|
// With autowrap off, by now we already know the width the label will take.
|
||||||
|
width = (minsize.width - style->get_minimum_size().width);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lines_dirty) {
|
||||||
|
BitField<TextServer::TextOverrunFlag> overrun_flags = TextServer::OVERRUN_NO_TRIM;
|
||||||
|
switch (overrun_behavior) {
|
||||||
|
case TextServer::OVERRUN_TRIM_WORD_ELLIPSIS:
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM_WORD_ONLY);
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_ADD_ELLIPSIS);
|
||||||
|
break;
|
||||||
|
case TextServer::OVERRUN_TRIM_ELLIPSIS:
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_ADD_ELLIPSIS);
|
||||||
|
break;
|
||||||
|
case TextServer::OVERRUN_TRIM_WORD:
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM_WORD_ONLY);
|
||||||
|
break;
|
||||||
|
case TextServer::OVERRUN_TRIM_CHAR:
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_TRIM);
|
||||||
|
break;
|
||||||
|
case TextServer::OVERRUN_NO_TRIMMING:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fill after min_size calculation.
|
||||||
|
|
||||||
|
int visible_lines = lines_rid.size();
|
||||||
|
if (max_lines_visible >= 0 && visible_lines > max_lines_visible) {
|
||||||
|
visible_lines = max_lines_visible;
|
||||||
|
}
|
||||||
|
if (autowrap_mode != TextServer::AUTOWRAP_OFF) {
|
||||||
|
bool lines_hidden = visible_lines > 0 && visible_lines < lines_rid.size();
|
||||||
|
if (lines_hidden) {
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_ENFORCE_ELLIPSIS);
|
||||||
|
}
|
||||||
|
if (horizontal_alignment == HORIZONTAL_ALIGNMENT_FILL) {
|
||||||
|
for (int i = 0; i < lines_rid.size(); i++) {
|
||||||
|
if (i < visible_lines - 1 || lines_rid.size() == 1) {
|
||||||
|
TS->shaped_text_fit_to_width(lines_rid[i], width);
|
||||||
|
} else if (i == (visible_lines - 1)) {
|
||||||
|
TS->shaped_text_overrun_trim_to_width(lines_rid[visible_lines - 1], width, overrun_flags);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (lines_hidden) {
|
||||||
|
TS->shaped_text_overrun_trim_to_width(lines_rid[visible_lines - 1], width, overrun_flags);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Autowrap disabled.
|
||||||
|
for (int i = 0; i < lines_rid.size(); i++) {
|
||||||
|
if (horizontal_alignment == HORIZONTAL_ALIGNMENT_FILL) {
|
||||||
|
TS->shaped_text_fit_to_width(lines_rid[i], width);
|
||||||
|
overrun_flags.set_flag(TextServer::OVERRUN_JUSTIFICATION_AWARE);
|
||||||
|
TS->shaped_text_overrun_trim_to_width(lines_rid[i], width, overrun_flags);
|
||||||
|
TS->shaped_text_fit_to_width(lines_rid[i], width, TextServer::JUSTIFICATION_WORD_BOUND | TextServer::JUSTIFICATION_KASHIDA | TextServer::JUSTIFICATION_CONSTRAIN_ELLIPSIS);
|
||||||
|
} else {
|
||||||
|
TS->shaped_text_overrun_trim_to_width(lines_rid[i], width, overrun_flags);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int last_line = MIN(lines_rid.size(), visible_lines + lines_skipped);
|
||||||
|
int line_spacing = settings.is_valid() ? settings->get_line_spacing() : theme_cache.line_spacing;
|
||||||
|
for (int64_t i = lines_skipped; i < last_line; i++) {
|
||||||
|
minsize.height += TS->shaped_text_get_size(lines_rid[i]).y + line_spacing;
|
||||||
|
}
|
||||||
|
|
||||||
|
lines_dirty = false;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Autowrap disabled.
|
callable_mp(this, &Label::_shape).call_deferred();
|
||||||
for (int i = 0; i < lines_rid.size(); i++) {
|
|
||||||
if (horizontal_alignment == HORIZONTAL_ALIGNMENT_FILL) {
|
|
||||||
TS->shaped_text_fit_to_width(lines_rid[i], width);
|
|
||||||
overrun_flags.set_flag(TextServer::OVERRUN_JUSTIFICATION_AWARE);
|
|
||||||
TS->shaped_text_overrun_trim_to_width(lines_rid[i], width, overrun_flags);
|
|
||||||
TS->shaped_text_fit_to_width(lines_rid[i], width, TextServer::JUSTIFICATION_WORD_BOUND | TextServer::JUSTIFICATION_KASHIDA | TextServer::JUSTIFICATION_CONSTRAIN_ELLIPSIS);
|
|
||||||
} else {
|
|
||||||
TS->shaped_text_overrun_trim_to_width(lines_rid[i], width, overrun_flags);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int last_line = MIN(lines_rid.size(), visible_lines + lines_skipped);
|
|
||||||
int line_spacing = settings.is_valid() ? settings->get_line_spacing() : theme_cache.line_spacing;
|
|
||||||
for (int64_t i = lines_skipped; i < last_line; i++) {
|
|
||||||
minsize.height += TS->shaped_text_get_size(lines_rid[i]).y + line_spacing;
|
|
||||||
}
|
|
||||||
|
|
||||||
lines_dirty = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (minsize != prev_minsize || font_was_dirty) {
|
if (draw_pending) {
|
||||||
|
queue_redraw();
|
||||||
|
draw_pending = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (minsize != prev_minsize) {
|
||||||
update_minimum_size();
|
update_minimum_size();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!width_stabilized) {
|
return can_process_lines;
|
||||||
callable_mp(this, &Label::_shape).call_deferred();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void draw_glyph(const Glyph &p_gl, const RID &p_canvas, const Color &p_font_color, const Vector2 &p_ofs) {
|
inline void draw_glyph(const Glyph &p_gl, const RID &p_canvas, const Color &p_font_color, const Vector2 &p_ofs) {
|
||||||
|
@ -367,10 +373,9 @@ void Label::_notification(int p_what) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dirty || font_dirty || lines_dirty) {
|
if (dirty || font_dirty || lines_dirty) {
|
||||||
_shape();
|
if (!_shape()) {
|
||||||
if (lines_dirty) {
|
|
||||||
// There will be another pass.
|
// There will be another pass.
|
||||||
queue_redraw();
|
draw_pending = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -396,7 +401,7 @@ void Label::_notification(int p_what) {
|
||||||
style->draw(ci, Rect2(Point2(0, 0), get_size()));
|
style->draw(ci, Rect2(Point2(0, 0), get_size()));
|
||||||
|
|
||||||
float total_h = 0.0;
|
float total_h = 0.0;
|
||||||
int visible_lines = 0;
|
int lines_visible = 0;
|
||||||
|
|
||||||
// Get number of lines to fit to the height.
|
// Get number of lines to fit to the height.
|
||||||
for (int64_t i = lines_skipped; i < lines_rid.size(); i++) {
|
for (int64_t i = lines_skipped; i < lines_rid.size(); i++) {
|
||||||
|
@ -404,14 +409,14 @@ void Label::_notification(int p_what) {
|
||||||
if (total_h > (get_size().height - style->get_minimum_size().height + line_spacing)) {
|
if (total_h > (get_size().height - style->get_minimum_size().height + line_spacing)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
visible_lines++;
|
lines_visible++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (max_lines_visible >= 0 && visible_lines > max_lines_visible) {
|
if (max_lines_visible >= 0 && lines_visible > max_lines_visible) {
|
||||||
visible_lines = max_lines_visible;
|
lines_visible = max_lines_visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
int last_line = MIN(lines_rid.size(), visible_lines + lines_skipped);
|
int last_line = MIN(lines_rid.size(), lines_visible + lines_skipped);
|
||||||
bool trim_chars = (visible_chars >= 0) && (visible_chars_behavior == TextServer::VC_CHARS_AFTER_SHAPING);
|
bool trim_chars = (visible_chars >= 0) && (visible_chars_behavior == TextServer::VC_CHARS_AFTER_SHAPING);
|
||||||
bool trim_glyphs_ltr = (visible_chars >= 0) && ((visible_chars_behavior == TextServer::VC_GLYPHS_LTR) || ((visible_chars_behavior == TextServer::VC_GLYPHS_AUTO) && !rtl_layout));
|
bool trim_glyphs_ltr = (visible_chars >= 0) && ((visible_chars_behavior == TextServer::VC_GLYPHS_LTR) || ((visible_chars_behavior == TextServer::VC_GLYPHS_AUTO) && !rtl_layout));
|
||||||
bool trim_glyphs_rtl = (visible_chars >= 0) && ((visible_chars_behavior == TextServer::VC_GLYPHS_RTL) || ((visible_chars_behavior == TextServer::VC_GLYPHS_AUTO) && rtl_layout));
|
bool trim_glyphs_rtl = (visible_chars >= 0) && ((visible_chars_behavior == TextServer::VC_GLYPHS_RTL) || ((visible_chars_behavior == TextServer::VC_GLYPHS_AUTO) && rtl_layout));
|
||||||
|
@ -428,7 +433,7 @@ void Label::_notification(int p_what) {
|
||||||
total_h += style->get_margin(SIDE_TOP) + style->get_margin(SIDE_BOTTOM);
|
total_h += style->get_margin(SIDE_TOP) + style->get_margin(SIDE_BOTTOM);
|
||||||
|
|
||||||
int vbegin = 0, vsep = 0;
|
int vbegin = 0, vsep = 0;
|
||||||
if (visible_lines > 0) {
|
if (lines_visible > 0) {
|
||||||
switch (vertical_alignment) {
|
switch (vertical_alignment) {
|
||||||
case VERTICAL_ALIGNMENT_TOP: {
|
case VERTICAL_ALIGNMENT_TOP: {
|
||||||
// Nothing.
|
// Nothing.
|
||||||
|
@ -445,8 +450,8 @@ void Label::_notification(int p_what) {
|
||||||
} break;
|
} break;
|
||||||
case VERTICAL_ALIGNMENT_FILL: {
|
case VERTICAL_ALIGNMENT_FILL: {
|
||||||
vbegin = 0;
|
vbegin = 0;
|
||||||
if (visible_lines > 1) {
|
if (lines_visible > 1) {
|
||||||
vsep = (size.y - (total_h - line_spacing)) / (visible_lines - 1);
|
vsep = (size.y - (total_h - line_spacing)) / (lines_visible - 1);
|
||||||
} else {
|
} else {
|
||||||
vsep = 0;
|
vsep = 0;
|
||||||
}
|
}
|
||||||
|
@ -615,6 +620,8 @@ void Label::_notification(int p_what) {
|
||||||
}
|
}
|
||||||
ofs.y += TS->shaped_text_get_descent(lines_rid[i]) + vsep + line_spacing;
|
ofs.y += TS->shaped_text_get_descent(lines_rid[i]) + vsep + line_spacing;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
draw_pending = false;
|
||||||
} break;
|
} break;
|
||||||
|
|
||||||
case NOTIFICATION_THEME_CHANGED: {
|
case NOTIFICATION_THEME_CHANGED: {
|
||||||
|
@ -666,25 +673,25 @@ int Label::get_line_count() const {
|
||||||
int Label::get_visible_line_count() const {
|
int Label::get_visible_line_count() const {
|
||||||
Ref<StyleBox> style = theme_cache.normal_style;
|
Ref<StyleBox> style = theme_cache.normal_style;
|
||||||
int line_spacing = settings.is_valid() ? settings->get_line_spacing() : theme_cache.line_spacing;
|
int line_spacing = settings.is_valid() ? settings->get_line_spacing() : theme_cache.line_spacing;
|
||||||
int visible_lines = 0;
|
int lines_visible = 0;
|
||||||
float total_h = 0.0;
|
float total_h = 0.0;
|
||||||
for (int64_t i = lines_skipped; i < lines_rid.size(); i++) {
|
for (int64_t i = lines_skipped; i < lines_rid.size(); i++) {
|
||||||
total_h += TS->shaped_text_get_size(lines_rid[i]).y + line_spacing;
|
total_h += TS->shaped_text_get_size(lines_rid[i]).y + line_spacing;
|
||||||
if (total_h > (get_size().height - style->get_minimum_size().height + line_spacing)) {
|
if (total_h > (get_size().height - style->get_minimum_size().height + line_spacing)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
visible_lines++;
|
lines_visible++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (visible_lines > lines_rid.size()) {
|
if (lines_visible > lines_rid.size()) {
|
||||||
visible_lines = lines_rid.size();
|
lines_visible = lines_rid.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (max_lines_visible >= 0 && visible_lines > max_lines_visible) {
|
if (max_lines_visible >= 0 && lines_visible > max_lines_visible) {
|
||||||
visible_lines = max_lines_visible;
|
lines_visible = max_lines_visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
return visible_lines;
|
return lines_visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Label::set_horizontal_alignment(HorizontalAlignment p_alignment) {
|
void Label::set_horizontal_alignment(HorizontalAlignment p_alignment) {
|
||||||
|
@ -969,7 +976,7 @@ void Label::_bind_methods() {
|
||||||
ClassDB::bind_method(D_METHOD("get_visible_ratio"), &Label::get_visible_ratio);
|
ClassDB::bind_method(D_METHOD("get_visible_ratio"), &Label::get_visible_ratio);
|
||||||
ClassDB::bind_method(D_METHOD("set_lines_skipped", "lines_skipped"), &Label::set_lines_skipped);
|
ClassDB::bind_method(D_METHOD("set_lines_skipped", "lines_skipped"), &Label::set_lines_skipped);
|
||||||
ClassDB::bind_method(D_METHOD("get_lines_skipped"), &Label::get_lines_skipped);
|
ClassDB::bind_method(D_METHOD("get_lines_skipped"), &Label::get_lines_skipped);
|
||||||
ClassDB::bind_method(D_METHOD("set_max_lines_visible", "visible_lines"), &Label::set_max_lines_visible);
|
ClassDB::bind_method(D_METHOD("set_max_lines_visible", "lines_visible"), &Label::set_max_lines_visible);
|
||||||
ClassDB::bind_method(D_METHOD("get_max_lines_visible"), &Label::get_max_lines_visible);
|
ClassDB::bind_method(D_METHOD("get_max_lines_visible"), &Label::get_max_lines_visible);
|
||||||
ClassDB::bind_method(D_METHOD("set_structured_text_bidi_override", "parser"), &Label::set_structured_text_bidi_override);
|
ClassDB::bind_method(D_METHOD("set_structured_text_bidi_override", "parser"), &Label::set_structured_text_bidi_override);
|
||||||
ClassDB::bind_method(D_METHOD("get_structured_text_bidi_override"), &Label::get_structured_text_bidi_override);
|
ClassDB::bind_method(D_METHOD("get_structured_text_bidi_override"), &Label::get_structured_text_bidi_override);
|
||||||
|
|
|
@ -50,7 +50,6 @@ private:
|
||||||
bool uppercase = false;
|
bool uppercase = false;
|
||||||
|
|
||||||
bool lines_dirty = true;
|
bool lines_dirty = true;
|
||||||
|
|
||||||
bool dirty = true;
|
bool dirty = true;
|
||||||
bool font_dirty = true;
|
bool font_dirty = true;
|
||||||
RID text_rid;
|
RID text_rid;
|
||||||
|
@ -66,6 +65,7 @@ private:
|
||||||
float visible_ratio = 1.0;
|
float visible_ratio = 1.0;
|
||||||
int lines_skipped = 0;
|
int lines_skipped = 0;
|
||||||
int max_lines_visible = -1;
|
int max_lines_visible = -1;
|
||||||
|
bool draw_pending = false;
|
||||||
|
|
||||||
Ref<LabelSettings> settings;
|
Ref<LabelSettings> settings;
|
||||||
|
|
||||||
|
@ -83,7 +83,7 @@ private:
|
||||||
int font_shadow_outline_size;
|
int font_shadow_outline_size;
|
||||||
} theme_cache;
|
} theme_cache;
|
||||||
|
|
||||||
void _shape();
|
bool _shape();
|
||||||
void _invalidate();
|
void _invalidate();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
Loading…
Add table
Reference in a new issue