Changes in uspace/lib/gfxfont/src/text.c [f2d4a46:a0aeb8f] in mainline
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/lib/gfxfont/src/text.c
rf2d4a46 ra0aeb8f 1 1 /* 2 * Copyright (c) 202 2Jiri Svoboda2 * Copyright (c) 2021 Jiri Svoboda 3 3 * All rights reserved. 4 4 * … … 97 97 /** Print string using text characters in text mode. 98 98 * 99 * @param font Font 99 100 * @param pos Position of top-left corner of text 100 * @param fmt Formatting101 * @param color Text color 101 102 * @param str String 102 103 * @return EOK on success or an error code 103 104 */ 104 static errno_t gfx_puttext_textmode(gfx_ coord2_t *pos, gfx_text_fmt_t *fmt,105 const char *str)106 { 107 gfx_context_t *gc = f mt->font->typeface->gc;105 static errno_t gfx_puttext_textmode(gfx_font_t *font, gfx_coord2_t *pos, 106 gfx_color_t *color, const char *str) 107 { 108 gfx_context_t *gc = font->typeface->gc; 108 109 gfx_bitmap_params_t params; 109 110 gfx_bitmap_t *bitmap; 110 111 gfx_bitmap_alloc_t alloc; 111 gfx_coord_t width;112 112 uint8_t attr; 113 113 pixelmap_t pmap; 114 114 gfx_coord_t x; 115 gfx_coord_t rmargin;116 115 pixel_t pixel; 117 116 char32_t c; 118 117 size_t off; 119 bool ellipsis;120 118 errno_t rc; 121 122 width = str_width(str);123 if (fmt->abbreviate && width > fmt->width) {124 ellipsis = true;125 width = fmt->width;126 if (width > 3)127 rmargin = width - 3;128 else129 rmargin = width;130 } else {131 ellipsis = false;132 rmargin = width;133 }134 119 135 120 /* … … 138 123 */ 139 124 140 gfx_color_get_ega( fmt->color, &attr);125 gfx_color_get_ega(color, &attr); 141 126 142 127 gfx_bitmap_params_init(¶ms); 143 128 params.rect.p0.x = 0; 144 129 params.rect.p0.y = 0; 145 params.rect.p1.x = width;130 params.rect.p1.x = str_width(str); 146 131 params.rect.p1.y = 1; 147 132 … … 166 151 167 152 off = 0; 168 for (x = 0; x < rmargin; x++) {153 for (x = 0; x < params.rect.p1.x; x++) { 169 154 c = str_decode(str, &off, STR_NO_LIMIT); 170 155 pixel = PIXEL(attr, … … 175 160 } 176 161 177 if (ellipsis) {178 for (x = rmargin; x < params.rect.p1.x; x++) {179 c = '.';180 pixel = PIXEL(attr,181 (c >> 16) & 0xff,182 (c >> 8) & 0xff,183 c & 0xff);184 pixelmap_put_pixel(&pmap, x, 0, pixel);185 }186 }187 188 162 rc = gfx_bitmap_render(bitmap, NULL, pos); 189 163 … … 194 168 /** Get text starting position. 195 169 * 170 * @param font Font 196 171 * @param pos Anchor position 197 172 * @param fmt Text formatting … … 199 174 * @param spos Place to store starting position 200 175 */ 201 void gfx_text_start_pos(gfx_ coord2_t *pos, gfx_text_fmt_t *fmt,202 const char *str, gfx_coord2_t *spos)176 void gfx_text_start_pos(gfx_font_t *font, gfx_coord2_t *pos, 177 gfx_text_fmt_t *fmt, const char *str, gfx_coord2_t *spos) 203 178 { 204 179 gfx_font_metrics_t fmetrics; … … 209 184 /* Adjust position for horizontal alignment */ 210 185 if (fmt->halign != gfx_halign_left) { 211 /* Compute text width */ 212 width = gfx_text_width(fmt->font, str); 213 if (fmt->abbreviate && width > fmt->width) 214 width = fmt->width; 215 186 width = gfx_text_width(font, str); 216 187 switch (fmt->halign) { 217 188 case gfx_halign_center: … … 227 198 228 199 /* Adjust position for vertical alignment */ 229 gfx_font_get_metrics(f mt->font, &fmetrics);200 gfx_font_get_metrics(font, &fmetrics); 230 201 231 202 if (fmt->valign != gfx_valign_baseline) { … … 248 219 /** Render text. 249 220 * 221 * @param font Font 250 222 * @param pos Anchor position 251 223 * @param fmt Text formatting … … 253 225 * @return EOK on success or an error code 254 226 */ 255 errno_t gfx_puttext(gfx_coord2_t *pos, gfx_text_fmt_t *fmt, const char *str) 227 errno_t gfx_puttext(gfx_font_t *font, gfx_coord2_t *pos, 228 gfx_text_fmt_t *fmt, const char *str) 256 229 { 257 230 gfx_glyph_metrics_t gmetrics; 258 gfx_font_metrics_t fmetrics;259 231 size_t stradv; 260 232 const char *cp; 261 233 gfx_glyph_t *glyph; 262 234 gfx_coord2_t cpos; 263 gfx_coord2_t spos;264 gfx_rect_t rect;265 gfx_coord_t width;266 gfx_coord_t rmargin;267 bool ellipsis;268 235 errno_t rc; 269 236 270 gfx_text_start_pos( pos, fmt, str, &spos);237 gfx_text_start_pos(font, pos, fmt, str, &cpos); 271 238 272 239 /* Text mode */ 273 if ((f mt->font->finfo->props.flags & gff_text_mode) != 0)274 return gfx_puttext_textmode( &spos, fmt, str);275 276 rc = gfx_set_color(f mt->font->typeface->gc, fmt->color);240 if ((font->finfo->props.flags & gff_text_mode) != 0) 241 return gfx_puttext_textmode(font, &cpos, fmt->color, str); 242 243 rc = gfx_set_color(font->typeface->gc, fmt->color); 277 244 if (rc != EOK) 278 245 return rc; 279 246 280 width = gfx_text_width(fmt->font, str);281 282 if (fmt->abbreviate && width > fmt->width) {283 /* Need to append ellipsis */284 ellipsis = true;285 rmargin = spos.x + fmt->width - gfx_text_width(fmt->font, "...");286 } else {287 ellipsis = false;288 rmargin = spos.x + width;289 }290 291 cpos = spos;292 247 cp = str; 293 248 while (*cp != '\0') { 294 rc = gfx_font_search_glyph(f mt->font, cp, &glyph, &stradv);249 rc = gfx_font_search_glyph(font, cp, &glyph, &stradv); 295 250 if (rc != EOK) { 296 251 ++cp; … … 300 255 gfx_glyph_get_metrics(glyph, &gmetrics); 301 256 302 /* Stop if we would run over the right margin */303 if (fmt->abbreviate && cpos.x + gmetrics.advance > rmargin)304 break;305 306 257 rc = gfx_glyph_render(glyph, &cpos); 307 258 if (rc != EOK) … … 312 263 } 313 264 314 /* Text underlining */315 if (fmt->underline) {316 gfx_font_get_metrics(fmt->font, &fmetrics);317 318 rect.p0.x = spos.x;319 rect.p0.y = spos.y + fmetrics.underline_y0;320 rect.p1.x = cpos.x;321 rect.p1.y = spos.y + fmetrics.underline_y1;322 323 rc = gfx_fill_rect(fmt->font->typeface->gc, &rect);324 if (rc != EOK)325 return rc;326 }327 328 /* Render ellipsis, if required */329 if (ellipsis) {330 rc = gfx_font_search_glyph(fmt->font, ".", &glyph, &stradv);331 if (rc != EOK)332 return EOK;333 334 gfx_glyph_get_metrics(glyph, &gmetrics);335 336 rc = gfx_glyph_render(glyph, &cpos);337 if (rc != EOK)338 return rc;339 340 cpos.x += gmetrics.advance;341 342 rc = gfx_glyph_render(glyph, &cpos);343 if (rc != EOK)344 return rc;345 346 cpos.x += gmetrics.advance;347 348 rc = gfx_glyph_render(glyph, &cpos);349 if (rc != EOK)350 return rc;351 }352 353 265 return EOK; 354 266 } … … 356 268 /** Find character position in string by X coordinate. 357 269 * 270 * @param font Font 358 271 * @param pos Anchor position 359 272 * @param fmt Text formatting … … 367 280 * offset of the following character. 368 281 */ 369 size_t gfx_text_find_pos(gfx_ coord2_t *pos, gfx_text_fmt_t *fmt,370 const char *str, gfx_coord2_t *fpos)282 size_t gfx_text_find_pos(gfx_font_t *font, gfx_coord2_t *pos, 283 gfx_text_fmt_t *fmt, const char *str, gfx_coord2_t *fpos) 371 284 { 372 285 gfx_glyph_metrics_t gmetrics; … … 379 292 errno_t rc; 380 293 381 gfx_text_start_pos( pos, fmt, str, &cpos);294 gfx_text_start_pos(font, pos, fmt, str, &cpos); 382 295 383 296 /* Text mode */ 384 if ((f mt->font->finfo->props.flags & gff_text_mode) != 0) {297 if ((font->finfo->props.flags & gff_text_mode) != 0) { 385 298 off = 0; 386 299 strsize = str_size(str); … … 398 311 off = 0; 399 312 while (*cp != '\0') { 400 rc = gfx_font_search_glyph(f mt->font, cp, &glyph, &stradv);313 rc = gfx_font_search_glyph(font, cp, &glyph, &stradv); 401 314 if (rc != EOK) { 402 315 ++cp; … … 424 337 * to the same objects, respectively. 425 338 * 339 * @param font Font 426 340 * @param pos Anchor position 427 341 * @param fmt Text formatting … … 430 344 * @param cfmt Place to store format for continuation 431 345 */ 432 void gfx_text_cont(gfx_coord2_t *pos, gfx_text_fmt_t *fmt, const char *str, 433 gfx_coord2_t *cpos, gfx_text_fmt_t *cfmt) 346 void gfx_text_cont(gfx_font_t *font, gfx_coord2_t *pos, 347 gfx_text_fmt_t *fmt, const char *str, gfx_coord2_t *cpos, 348 gfx_text_fmt_t *cfmt) 434 349 { 435 350 gfx_coord2_t spos; … … 437 352 438 353 /* Continuation should start where the current string ends */ 439 gfx_text_start_pos( pos, fmt, str, &spos);440 cpos->x = spos.x + gfx_text_width(f mt->font, str);354 gfx_text_start_pos(font, pos, fmt, str, &spos); 355 cpos->x = spos.x + gfx_text_width(font, str); 441 356 cpos->y = spos.y; 442 357 … … 449 364 tfmt.valign = gfx_valign_baseline; 450 365 451 /* Remaining available width */452 tfmt.width = fmt->width - (cpos->x - spos.x);453 454 366 *cfmt = tfmt; 455 367 } … … 457 369 /** Get text bounding rectangle. 458 370 * 371 * @param font Font 459 372 * @param pos Anchor position 460 373 * @param fmt Text formatting … … 462 375 * @param rect Place to store bounding rectangle 463 376 */ 464 void gfx_text_rect(gfx_ coord2_t *pos, gfx_text_fmt_t *fmt, const char *str,465 gfx_ rect_t *rect)377 void gfx_text_rect(gfx_font_t *font, gfx_coord2_t *pos, 378 gfx_text_fmt_t *fmt, const char *str, gfx_rect_t *rect) 466 379 { 467 380 gfx_coord2_t spos; 468 gfx_coord_t width; 469 470 gfx_text_start_pos(pos, fmt, str, &spos); 471 width = gfx_text_width(fmt->font, str); 472 if (fmt->abbreviate && width > fmt->width) 473 width = fmt->width; 381 382 gfx_text_start_pos(font, pos, fmt, str, &spos); 474 383 475 384 rect->p0.x = spos.x; 476 rect->p0.y = spos.y - f mt->font->metrics.ascent;477 rect->p1.x = spos.x + width;478 rect->p1.y = spos.y + fmt->font->metrics.descent + 1;385 rect->p0.y = spos.y - font->metrics.ascent; 386 rect->p1.x = spos.x + gfx_text_width(font, str); 387 rect->p1.y = spos.y + font->metrics.descent + 1; 479 388 } 480 389
Note:
See TracChangeset
for help on using the changeset viewer.