X-Git-Url: https://git.artsoft.org/?a=blobdiff_plain;f=src%2Flibgame%2Ftext.c;h=132016de40c9d8d1e2fbd20d23640ba9d4853458;hb=74c0f7de91268e40d15948f473eac51a9760b9c0;hp=f41635ae3f4863b659f9283a7238bcb761cda5d3;hpb=2ffc7d13be24ed07f4dcb9a77924450f16597720;p=rocksndiamonds.git diff --git a/src/libgame/text.c b/src/libgame/text.c index f41635ae..132016de 100644 --- a/src/libgame/text.c +++ b/src/libgame/text.c @@ -50,7 +50,7 @@ static void InitFontClipmasks() clip_gc_valuemask, &clip_gc_values); /* create only those clipping Pixmaps we really need */ - for (i=0; i < gfx.num_fonts; i++) + for (i = 0; i < gfx.num_fonts; i++) { if (gfx.font_bitmap_info[i].bitmap == NULL) continue; @@ -58,7 +58,7 @@ static void InitFontClipmasks() gfx.font_bitmap_info[i].clip_mask = checked_calloc(gfx.font_bitmap_info[i].num_chars * sizeof(Pixmap)); - for (j=0; j < gfx.font_bitmap_info[i].num_chars; j++) + for (j = 0; j < gfx.font_bitmap_info[i].num_chars; j++) { Bitmap *src_bitmap = gfx.font_bitmap_info[i].bitmap; Pixmap src_pixmap = src_bitmap->clip_mask; @@ -87,11 +87,11 @@ static void FreeFontClipmasks() if (gfx.num_fonts == 0 || gfx.font_bitmap_info[0].bitmap == NULL) return; - for (i=0; i < gfx.num_fonts; i++) + for (i = 0; i < gfx.num_fonts; i++) { if (gfx.font_bitmap_info[i].clip_mask) { - for (j=0; j < gfx.font_bitmap_info[i].num_chars; j++) + for (j = 0; j < gfx.font_bitmap_info[i].num_chars; j++) XFreePixmap(display, gfx.font_bitmap_info[i].clip_mask[j]); free(gfx.font_bitmap_info[i].clip_mask); } @@ -191,7 +191,7 @@ void DrawInitText(char *text, int ypos, int font_nr) } } -void DrawTextFCentered(int y, int font_nr, char *format, ...) +void DrawTextF(int x, int y, int font_nr, char *format, ...) { char buffer[MAX_OUTPUT_LINESIZE + 1]; va_list ap; @@ -201,13 +201,12 @@ void DrawTextFCentered(int y, int font_nr, char *format, ...) va_end(ap); if (strlen(buffer) > MAX_OUTPUT_LINESIZE) - Error(ERR_EXIT, "string too long in DrawTextFCentered() -- aborting"); + Error(ERR_EXIT, "string too long in DrawTextF() -- aborting"); - DrawText(gfx.sx + (gfx.sxsize - getTextWidth(buffer, font_nr)) / 2, - gfx.sy + y, buffer, font_nr); + DrawText(gfx.sx + x, gfx.sy + y, buffer, font_nr); } -void DrawTextF(int x, int y, int font_nr, char *format, ...) +void DrawTextFCentered(int y, int font_nr, char *format, ...) { char buffer[MAX_OUTPUT_LINESIZE + 1]; va_list ap; @@ -217,9 +216,21 @@ void DrawTextF(int x, int y, int font_nr, char *format, ...) va_end(ap); if (strlen(buffer) > MAX_OUTPUT_LINESIZE) - Error(ERR_EXIT, "string too long in DrawTextF() -- aborting"); + Error(ERR_EXIT, "string too long in DrawTextFCentered() -- aborting"); - DrawText(gfx.sx + x, gfx.sy + y, buffer, font_nr); + DrawText(gfx.sx + (gfx.sxsize - getTextWidth(buffer, font_nr)) / 2, + gfx.sy + y, buffer, font_nr); +} + +void DrawTextS(int x, int y, int font_nr, char *text) +{ + DrawText(gfx.sx + x, gfx.sy + y, text, font_nr); +} + +void DrawTextSCentered(int y, int font_nr, char *text) +{ + DrawText(gfx.sx + (gfx.sxsize - getTextWidth(text, font_nr)) / 2, + gfx.sy + y, text, font_nr); } void DrawText(int x, int y, char *text, int font_nr) @@ -340,7 +351,7 @@ void DrawTextToTextArea(int x, int y, char *text, int font_nr, int line_length, char buffer[MAX_OUTPUT_LINESIZE + 1]; int i; - for (i=0; i < line_length && *text; i++) + for (i = 0; i < line_length && *text; i++) if ((buffer[i] = *text++) == '\n') break; buffer[MIN(i, area_xsize)] = '\0'; @@ -353,3 +364,103 @@ void DrawTextToTextArea(int x, int y, char *text, int font_nr, int line_length, redraw_mask |= REDRAW_FIELD; } + +boolean RenderLineToBuffer(char **src_buffer_ptr, char *dst_buffer, + int *dst_buffer_len, boolean last_line_was_empty, + int max_chars_per_line) +{ + char *text_ptr = *src_buffer_ptr; + char *buffer = dst_buffer; + int buffer_len = *dst_buffer_len; + boolean buffer_filled = FALSE; + + while (*text_ptr) + { + char *word_ptr; + int word_len; + + /* skip leading whitespaces */ + while (*text_ptr == ' ' || *text_ptr == '\t') + text_ptr++; + + word_ptr = text_ptr; + word_len = 0; + + /* look for end of next word */ + while (*word_ptr != ' ' && *word_ptr != '\t' && *word_ptr != '\0') + { + word_ptr++; + word_len++; + } + + if (word_len == 0) + { + continue; + } + else if (*text_ptr == '\n') /* special case: force empty line */ + { + if (buffer_len == 0) + text_ptr++; + + /* prevent printing of multiple empty lines */ + if (buffer_len > 0 || !last_line_was_empty) + buffer_filled = TRUE; + } + else if (word_len < max_chars_per_line - buffer_len) + { + /* word fits into text buffer -- add word */ + + if (buffer_len > 0) + buffer[buffer_len++] = ' '; + + strncpy(&buffer[buffer_len], text_ptr, word_len); + buffer_len += word_len; + buffer[buffer_len] = '\0'; + text_ptr += word_len; + } + else if (buffer_len > 0) + { + /* not enough space left for word in text buffer -- print buffer */ + + buffer_filled = TRUE; + } + else + { + /* word does not fit at all into empty text buffer -- cut word */ + + strncpy(buffer, text_ptr, max_chars_per_line); + buffer[max_chars_per_line] = '\0'; + text_ptr += max_chars_per_line; + buffer_filled = TRUE; + } + + if (buffer_filled) + break; + } + + *src_buffer_ptr = text_ptr; + *dst_buffer_len = buffer_len; + + return buffer_filled; +} + +void DrawTextWrapped(int x, int y, char *text, int font_nr, int line_length, + int max_lines) +{ + char *text_ptr = text; + char buffer[line_length + 1]; + int buffer_len; + int current_line = 0; + int font_height = getFontHeight(font_nr); + + while (*text_ptr && current_line < max_lines) + { + buffer[0] = '\0'; + buffer_len = 0; + + RenderLineToBuffer(&text_ptr, buffer, &buffer_len, TRUE, line_length); + + DrawText(x, y + current_line * font_height, buffer, font_nr); + current_line++; + } +}