jlabel: align top by default and wrap lines in a smarter way

With this change, unless wrapped_newline_spadcing is set (which it's not
by default), spaces around explicit newlines will no longer be skipped.
This commit is contained in:
Lephenixnoir 2024-09-14 12:20:13 +02:00
parent 33e9962209
commit 4728c6ecbe
No known key found for this signature in database
GPG key ID: 1BBA026E13FC0495
2 changed files with 26 additions and 11 deletions

View file

@ -76,12 +76,16 @@ typedef struct {
of line number n */
DECLARE_VEC(uint16_t, breaks);
/* Block width (maximum length of a rendered line) */
uint16_t block_width;
/* Text wrapping mode */
enum jwrapmode wrap_mode;
/* Whether the text has been allocated by the label or supplied by user */
int8_t owns_text;
/* Block width (maximum length of a rendered line) */
uint16_t block_width;
bool owns_text :1;
/* Whether to preserve spaces around line-wrapping newlines */
bool wrapped_newline_spacing :1;
uint :5;
/* Color and font of text; if NULL, gint's default font is used */
int color;
@ -130,5 +134,6 @@ void jlabel_set_line_spacing(jlabel *l, int line_spacing);
void jlabel_set_wrap_mode(jlabel *l, jwrapmode mode);
void jlabel_set_text_color(jlabel *l, int color);
void jlabel_set_font(jlabel *l, font_t const *font);
void jlabel_set_wrapped_newline_spacing(jlabel *l, bool preserve);
#endif /* _J_JLABEL */

View file

@ -21,7 +21,7 @@ jlabel *jlabel_create(char const *text, void *parent)
jwidget_init(&l->widget, jlabel_type_id, parent);
l->block_halign = J_ALIGN_LEFT;
l->block_valign = J_ALIGN_MIDDLE;
l->block_valign = J_ALIGN_TOP;
l->text_align = J_ALIGN_LEFT;
l->line_spacing = 0;
@ -31,6 +31,7 @@ jlabel *jlabel_create(char const *text, void *parent)
l->wrap_mode = J_WRAP_NONE;
l->text = text;
l->owns_text = false;
l->wrapped_newline_spacing = false;
vec_init(&l->breaks_vec, sizeof *l->breaks);
@ -188,6 +189,12 @@ void jlabel_set_font(jlabel *l, font_t const *font)
l->widget.dirty = 1;
}
void jlabel_set_wrapped_newline_spacing(jlabel *l, bool preserve)
{
l->wrapped_newline_spacing = preserve;
l->widget.dirty = 1;
}
//---
// Polymorphic widget operations
//---
@ -236,7 +243,7 @@ static void jlabel_poly_layout(void *l0)
/* Start of line */
add_break(l, str - l->text);
/* A "\n" forces a newline in all wrap omdes */
/* A "\n" forces a newline in all wrap modes */
char const *end_of_line = strchrnul(str, '\n');
/* Also consider word or letters boundaries */
@ -257,14 +264,17 @@ static void jlabel_poly_layout(void *l0)
}
}
bool natural_break = (*end_of_line == '\n');
char const *next_start = end_of_line + (*end_of_line == '\n');
/* Skip trailing spaces on this line */
while(end_of_line > str && end_of_line[-1] == ' ')
end_of_line--;
/* Skip leading spaces on the next line */
while(next_start[0] == ' ')
next_start++;
if(!natural_break && !l->wrapped_newline_spacing) {
/* Skip trailing spaces on this line */
while(end_of_line > str && end_of_line[-1] == ' ')
end_of_line--;
/* Skip leading spaces on the next line */
while(next_start[0] == ' ')
next_start++;
}
add_break(l, end_of_line - l->text);