Changeset 2cc1ec0 in mainline
- Timestamp:
- 2014-08-27T21:23:01Z (10 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- a9763c6
- Parents:
- 613d644
- Location:
- uspace/lib
- Files:
-
- 2 added
- 13 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/lib/draw/Makefile
r613d644 r2cc1ec0 37 37 cursor/embedded.c \ 38 38 font/embedded.c \ 39 font/bitmap_backend.c \ 39 40 gfx/font-8x16.c \ 40 41 gfx/cursor-11x18.c \ -
uspace/lib/draw/font.c
r613d644 r2cc1ec0 1 1 /* 2 2 * Copyright (c) 2012 Petr Koupy 3 * Copyright (c) 2014 Martin Sucha 3 4 * All rights reserved. 4 5 * … … 34 35 */ 35 36 36 #include <assert.h>37 37 #include <malloc.h> 38 #include <errno.h> 39 #include <str.h> 38 40 39 41 #include "font.h" … … 41 43 #include "drawctx.h" 42 44 43 void font_init(font_t *font, font_decoder_type_t decoder, char *path, uint16_t points)45 font_t *font_create(font_backend_t *backend, void *backend_data) 44 46 { 45 font->points = points; 46 47 switch (decoder) { 48 case FONT_DECODER_EMBEDDED: 49 font->decoder = &fd_embedded; 50 break; 51 default: 52 font->decoder = NULL; 53 break; 54 } 55 56 if (font->decoder) { 57 font->decoder->init(path, &font->glyph_count, &font->decoder_data); 58 59 if (font->glyph_count > 0) { 60 font->glyphs = (surface_t **) malloc(sizeof(surface_t *) * font->glyph_count); 61 } else { 62 font->glyphs = NULL; 63 } 64 65 if (font->glyphs) { 66 for (size_t i = 0; i < font->glyph_count; ++i) { 67 font->glyphs[i] = NULL; 68 } 69 } else { 70 font->glyph_count = 0; 71 } 72 } else { 73 font->glyph_count = 0; 74 font->glyphs = NULL; 75 font->decoder_data = NULL; 76 } 47 font_t *font = malloc(sizeof(font_t)); 48 if (font == NULL) 49 return NULL; 50 51 font->backend = backend; 52 font->backend_data = backend_data; 53 54 return font; 77 55 } 78 56 79 57 void font_release(font_t *font) 80 58 { 81 if (font->glyphs) { 82 for (size_t i = 0; i < font->glyph_count; ++i) { 83 if (font->glyphs[i]) { 84 surface_destroy(font->glyphs[i]); 59 font->backend->release(font->backend_data); 60 } 61 62 int font_get_metrics(font_t *font, font_metrics_t *metrics) { 63 return font->backend->get_font_metrics(font->backend_data, metrics); 64 } 65 66 int font_resolve_glyph(font_t *font, wchar_t c, glyph_id_t *glyph_id) { 67 return font->backend->resolve_glyph(font->backend_data, c, glyph_id); 68 } 69 70 int font_get_glyph_metrics(font_t *font, glyph_id_t glyph_id, 71 glyph_metrics_t *glyph_metrics) 72 { 73 return font->backend->get_glyph_metrics(font->backend_data, 74 glyph_id, glyph_metrics); 75 } 76 77 int font_render_glyph(font_t *font, drawctx_t *context, source_t *source, 78 sysarg_t x, sysarg_t y, glyph_id_t glyph_id) 79 { 80 return font->backend->render_glyph(font->backend_data, context, source, 81 x, y, glyph_id); 82 } 83 84 /* TODO this is bad interface */ 85 int font_get_box(font_t *font, char *text, sysarg_t *width, sysarg_t *height) 86 { 87 font_metrics_t fm; 88 int rc = font_get_metrics(font, &fm); 89 if (rc != EOK) 90 return rc; 91 92 native_t x = 0; 93 94 size_t off = 0; 95 while (true) { 96 wchar_t c = str_decode(text, &off, STR_NO_LIMIT); 97 if (c == 0) 98 break; 99 100 glyph_id_t glyph_id; 101 rc = font_resolve_glyph(font, c, &glyph_id); 102 if (rc != EOK) { 103 int rc2 = font_resolve_glyph(font, U_SPECIAL, &glyph_id); 104 if (rc2 != EOK) { 105 return rc; 85 106 } 86 107 } 87 free(font->glyphs); 108 109 glyph_metrics_t glyph_metrics; 110 rc = font_get_glyph_metrics(font, glyph_id, &glyph_metrics); 111 if (rc != EOK) 112 return rc; 113 114 x += glyph_metrics_get_advancement(&glyph_metrics); 88 115 } 89 90 if (font->decoder) {91 font->decoder->release(font->decoder_data);92 }116 117 *width = x; 118 *height = fm.ascender + fm.descender; 119 return EOK; 93 120 } 94 121 95 void font_get_box(font_t *font, char *text, sysarg_t *width, sysarg_t *height) 122 /* TODO this is bad interface */ 123 int font_draw_text(font_t *font, drawctx_t *context, source_t *source, 124 const char *text, sysarg_t sx, sysarg_t sy) 96 125 { 97 assert(width);98 assert(height);99 100 (*width) = 0;101 (*height) = 0;102 103 if (!text) {104 return;105 }106 107 while (*text) {108 uint16_t glyph_idx = font->decoder->resolve(*text, font->decoder_data);109 if (glyph_idx < font->glyph_count) {110 if (!font->glyphs[glyph_idx]) {111 font->glyphs[glyph_idx] =112 font->decoder->render(glyph_idx, font->points);113 }114 115 surface_t *glyph = font->glyphs[glyph_idx];116 if (glyph) {117 sysarg_t w;118 sysarg_t h;119 surface_get_resolution(glyph, &w, &h);120 (*width) += w;121 (*height) = (*height) < h ? h : (*height);122 }123 }124 ++text;125 }126 }127 128 void font_draw_text(font_t *font, drawctx_t *context, source_t *source,129 const char *text, sysarg_t x, sysarg_t y)130 {131 assert(context);132 assert(source);133 134 126 drawctx_save(context); 135 127 drawctx_set_compose(context, compose_over); 136 128 137 while (*text) { 138 uint16_t glyph_idx = font->decoder->resolve(*text, font->decoder_data); 139 if (glyph_idx < font->glyph_count) { 140 if (!font->glyphs[glyph_idx]) { 141 font->glyphs[glyph_idx] = 142 font->decoder->render(glyph_idx, font->points); 143 } 129 font_metrics_t fm; 130 int rc = font_get_metrics(font, &fm); 131 if (rc != EOK) 132 return rc; 144 133 145 surface_t *glyph = font->glyphs[glyph_idx]; 146 if (glyph) { 147 sysarg_t w; 148 sysarg_t h; 149 surface_get_resolution(glyph, &w, &h); 134 native_t baseline = sy + fm.ascender; 135 native_t x = sx; 150 136 151 transform_t transform; 152 transform_identity(&transform); 153 transform_translate(&transform, x, y); 154 source_set_transform(source, transform); 155 source_set_mask(source, glyph, false); 156 drawctx_transfer(context, x, y, w, h); 157 158 x += w; 137 size_t off = 0; 138 while (true) { 139 wchar_t c = str_decode(text, &off, STR_NO_LIMIT); 140 if (c == 0) 141 break; 142 143 glyph_id_t glyph_id; 144 rc = font_resolve_glyph(font, c, &glyph_id); 145 if (rc != EOK) { 146 int rc2 = font_resolve_glyph(font, U_SPECIAL, &glyph_id); 147 if (rc2 != EOK) { 148 return rc; 159 149 } 160 150 } 161 ++text; 151 152 glyph_metrics_t glyph_metrics; 153 rc = font_get_glyph_metrics(font, glyph_id, &glyph_metrics); 154 if (rc != EOK) 155 return rc; 156 157 rc = font_render_glyph(font, context, source, x, baseline, 158 glyph_id); 159 if (rc != EOK) 160 return rc; 161 162 x += glyph_metrics_get_advancement(&glyph_metrics); 163 162 164 } 163 165 164 166 drawctx_restore(context); 165 167 source_set_mask(source, NULL, false); 168 169 return EOK; 166 170 } 167 171 -
uspace/lib/draw/font.h
r613d644 r2cc1ec0 1 1 /* 2 2 * Copyright (c) 2012 Petr Koupy 3 * Copyright (c) 2014 Martin Sucha 3 4 * All rights reserved. 4 5 * … … 45 46 typedef struct drawctx drawctx_t; 46 47 47 typedef enum { 48 FONT_DECODER_EMBEDDED 49 } font_decoder_type_t; 48 typedef int metric_t; 50 49 51 50 typedef struct { 52 void (*init)(char *, uint16_t *, void **); 53 uint16_t (*resolve)(const wchar_t, void *); 54 surface_t *(*render)(uint16_t, uint16_t); 51 /* Horizontal distance between origin and left side of the glyph */ 52 metric_t left_side_bearing; 53 54 /* Width of the actual glyph drawn */ 55 metric_t width; 56 57 /* Horizontal distance between right side of the glyph and origin 58 of the next glyph */ 59 metric_t right_side_bearing; 60 61 /* Vertical distance between baseline and top of the glyph 62 (positive to top) */ 63 metric_t ascender; 64 65 /* Height of the actual glyph drawn */ 66 metric_t height; 67 } glyph_metrics_t; 68 69 static inline metric_t glyph_metrics_get_descender(glyph_metrics_t *gm) 70 { 71 return gm->height - gm->ascender; 72 } 73 74 static inline metric_t glyph_metrics_get_advancement(glyph_metrics_t *gm) 75 { 76 return gm->left_side_bearing + gm->width + gm->right_side_bearing; 77 } 78 79 typedef struct { 80 /* Distance between top of the line and baseline */ 81 metric_t ascender; 82 83 /* Distance between baseline and bottom of the line */ 84 metric_t descender; 85 86 /* Distance between bottom of the line and top of the next line */ 87 metric_t leading; 88 } font_metrics_t; 89 90 typedef uint32_t glyph_id_t; 91 92 typedef struct { 93 int (*get_font_metrics)(void *, font_metrics_t *); 94 int (*resolve_glyph)(void *, wchar_t, glyph_id_t *); 95 int (*get_glyph_metrics)(void *, glyph_id_t, glyph_metrics_t *); 96 int (*render_glyph)(void *, drawctx_t *, source_t *, sysarg_t, 97 sysarg_t, glyph_id_t); 55 98 void (*release)(void *); 56 } font_ decoder_t;99 } font_backend_t; 57 100 58 typedef struct font { 59 uint16_t points; 60 uint16_t glyph_count; 61 surface_t **glyphs; 62 font_decoder_t *decoder; 63 void *decoder_data; 101 typedef struct { 102 font_backend_t *backend; 103 void *backend_data; 64 104 } font_t; 65 105 66 extern void font_init(font_t *, font_decoder_type_t, char *, uint16_t); 106 extern font_t *font_create(font_backend_t *, void *); 107 extern int font_get_metrics(font_t *, font_metrics_t *); 108 extern int font_resolve_glyph(font_t *, wchar_t, glyph_id_t *); 109 extern int font_get_glyph_metrics(font_t *, glyph_id_t, glyph_metrics_t *); 110 extern int font_render_glyph(font_t *, drawctx_t *, source_t *, 111 sysarg_t, sysarg_t, glyph_id_t); 67 112 extern void font_release(font_t *); 68 113 69 extern voidfont_get_box(font_t *, char *, sysarg_t *, sysarg_t *);70 extern voidfont_draw_text(font_t *, drawctx_t *, source_t *, const char *,114 extern int font_get_box(font_t *, char *, sysarg_t *, sysarg_t *); 115 extern int font_draw_text(font_t *, drawctx_t *, source_t *, const char *, 71 116 sysarg_t, sysarg_t); 72 117 -
uspace/lib/draw/font/embedded.c
r613d644 r2cc1ec0 1 1 /* 2 2 * Copyright (c) 2012 Petr Koupy 3 * Copyright (c) 2014 Martin Sucha 3 4 * All rights reserved. 4 5 * … … 34 35 */ 35 36 36 #include <assert.h>37 37 #include <sys/types.h> 38 38 #include <malloc.h> 39 #include <errno.h> 39 40 40 41 #include "../gfx/font-8x16.h" 41 42 #include "embedded.h" 42 43 #include "../drawctx.h" 44 #include "bitmap_backend.h" 43 45 44 static void fde_init(char *path, uint16_t *glyph_count, void **data) 46 static int fde_resolve_glyph(void *unused, const wchar_t chr, 47 glyph_id_t *glyph_id) 45 48 { 46 assert(glyph_count); 47 assert(data); 48 49 (*glyph_count) = FONT_GLYPHS; 50 (*data) = NULL; 49 bool found = false; 50 uint16_t glyph = fb_font_glyph(chr, &found); 51 if (!found) 52 return ENOENT; 53 54 *glyph_id = glyph; 55 return EOK; 51 56 } 52 57 53 static uint16_t fde_resolve(const wchar_t chr, void *data) 58 static int fde_load_glyph_surface(void *unused, glyph_id_t glyph_id, 59 surface_t **out_surface) 54 60 { 55 return fb_font_glyph(chr); 61 surface_t *surface = surface_create(FONT_WIDTH, FONT_SCANLINES, NULL, 0); 62 if (!surface) 63 return ENOMEM; 64 65 for (unsigned int y = 0; y < FONT_SCANLINES; ++y) { 66 for (unsigned int x = 0; x < FONT_WIDTH; ++x) { 67 pixel_t p = (fb_font[glyph_id][y] & (1 << (7 - x))) ? 68 PIXEL(255, 0, 0, 0) : PIXEL(0, 0, 0, 0); 69 surface_put_pixel(surface, x, y, p); 70 } 71 } 72 73 *out_surface = surface; 74 return EOK; 56 75 } 57 76 58 static surface_t *fde_render(uint16_t glyph, uint16_t points) 77 static int fde_load_glyph_metrics(void *unused, glyph_id_t glyph_id, 78 glyph_metrics_t *gm) 59 79 { 60 surface_t *template = surface_create(FONT_WIDTH, FONT_SCANLINES, NULL, 0); 61 if (!template) { 62 return NULL; 63 } 64 for (unsigned int y = 0; y < FONT_SCANLINES; ++y) { 65 for (unsigned int x = 0; x < FONT_WIDTH; ++x) { 66 pixel_t p = (fb_font[glyph][y] & (1 << (7 - x))) ? 67 PIXEL(255, 0, 0, 0) : PIXEL(0, 0, 0, 0); 68 surface_put_pixel(template, x, y, p); 69 } 70 } 71 72 source_t source; 73 source_init(&source); 74 source_set_texture(&source, template, false); 75 76 transform_t transform; 77 transform_identity(&transform); 78 if (points != FONT_SCANLINES) { 79 double ratio = ((double) points) / ((double) FONT_SCANLINES); 80 transform_scale(&transform, ratio, ratio); 81 source_set_transform(&source, transform); 82 } 83 84 double width = FONT_WIDTH; 85 double height = FONT_SCANLINES; 86 transform_apply_linear(&transform, &width, &height); 87 surface_t *result = 88 surface_create((sysarg_t) (width + 0.5), (sysarg_t) (height + 0.5), NULL, 0); 89 if (!result) { 90 surface_destroy(template); 91 return NULL; 92 } 93 94 drawctx_t context; 95 drawctx_init(&context, result); 96 drawctx_set_source(&context, &source); 97 drawctx_transfer(&context, 0, 0, 98 (sysarg_t) (width + 0.5), (sysarg_t) (height + 0.5)); 99 100 surface_destroy(template); 101 102 return result; 80 /* This is simple monospaced font, so fill this data statically */ 81 gm->left_side_bearing = 0; 82 gm->width = FONT_WIDTH; 83 gm->right_side_bearing = 0; 84 gm->ascender = FONT_ASCENDER; 85 gm->height = FONT_SCANLINES; 86 87 return EOK; 103 88 } 104 89 … … 108 93 } 109 94 110 font_decoder_t fd_embedded = {111 . init = fde_init,112 . resolve = fde_resolve,113 . render = fde_render,95 bitmap_font_decoder_t fd_embedded = { 96 .resolve_glyph = fde_resolve_glyph, 97 .load_glyph_surface = fde_load_glyph_surface, 98 .load_glyph_metrics = fde_load_glyph_metrics, 114 99 .release = fde_release 115 100 }; 116 101 102 font_metrics_t font_metrics = { 103 .ascender = FONT_ASCENDER, 104 .descender = (FONT_SCANLINES - FONT_ASCENDER), 105 .leading = 0 106 }; 107 108 int embedded_font_create(font_t **font, uint16_t points) 109 { 110 return bitmap_font_create(&fd_embedded, NULL, FONT_GLYPHS, font_metrics, 111 points, font); 112 } 113 117 114 /** @} 118 115 */ -
uspace/lib/draw/font/embedded.h
r613d644 r2cc1ec0 39 39 #include "../font.h" 40 40 41 extern font_decoder_t fd_embedded;41 extern int embedded_font_create(font_t **, uint16_t points); 42 42 43 43 #endif -
uspace/lib/draw/gfx/font-8x16.c
r613d644 r2cc1ec0 44 44 * mark glyph if no specific glyph exists. 45 45 * 46 * If found is not null, indicate whether the glyph was found or not. 47 * 46 48 */ 47 uint16_t fb_font_glyph(const wchar_t ch )49 uint16_t fb_font_glyph(const wchar_t ch, bool *found) 48 50 { 51 if (found) 52 *found = true; 53 49 54 if (ch == 0x0000) 50 55 return 0; … … 361 366 if (ch == 0xfeff) 362 367 return 2896; 368 369 if (found) 370 *found = false; 363 371 364 372 return 2898; -
uspace/lib/draw/gfx/font-8x16.h
r613d644 r2cc1ec0 37 37 38 38 #include <sys/types.h> 39 #include <stdbool.h> 39 40 40 41 #define FONT_GLYPHS 2899 41 42 #define FONT_WIDTH 8 42 43 #define FONT_SCANLINES 16 44 #define FONT_ASCENDER 12 43 45 44 extern uint16_t fb_font_glyph(const wchar_t );46 extern uint16_t fb_font_glyph(const wchar_t, bool *); 45 47 extern uint8_t fb_font[FONT_GLYPHS][FONT_SCANLINES]; 46 48 -
uspace/lib/gui/button.c
r613d644 r2cc1ec0 38 38 #include <drawctx.h> 39 39 #include <surface.h> 40 #include <font/embedded.h> 41 #include <errno.h> 40 42 #include "common.h" 41 43 #include "window.h" … … 76 78 sysarg_t cpt_width; 77 79 sysarg_t cpt_height; 78 font_get_box( &btn->font, btn->caption, &cpt_width, &cpt_height);80 font_get_box(btn->font, btn->caption, &cpt_width, &cpt_height); 79 81 80 82 if ((widget->width >= cpt_width) && (widget->height >= cpt_height)) { … … 83 85 84 86 drawctx_set_source(&drawctx, &btn->text); 85 drawctx_set_font(&drawctx, &btn->font);87 drawctx_set_font(&drawctx, btn->font); 86 88 87 89 if (btn->caption) … … 96 98 widget_deinit(&btn->widget); 97 99 free(btn->caption); 98 font_release( &btn->font);100 font_release(btn->font); 99 101 } 100 102 … … 171 173 btn->caption = str_dup(caption); 172 174 173 font_init(&btn->font, FONT_DECODER_EMBEDDED, NULL, points); 175 int rc = embedded_font_create(&btn->font, points); 176 if (rc != EOK) { 177 free(btn->caption); 178 btn->caption = NULL; 179 return false; 180 } 174 181 175 182 sysarg_t cpt_width; 176 183 sysarg_t cpt_height; 177 font_get_box( &btn->font, btn->caption, &cpt_width, &cpt_height);184 font_get_box(btn->font, btn->caption, &cpt_width, &cpt_height); 178 185 btn->widget.width_min = cpt_width + 10; 179 186 btn->widget.height_min = cpt_height + 10; -
uspace/lib/gui/button.h
r613d644 r2cc1ec0 52 52 source_t text; 53 53 char *caption; 54 font_t font;54 font_t *font; 55 55 signal_t clicked; 56 56 } button_t; -
uspace/lib/gui/label.c
r613d644 r2cc1ec0 38 38 #include <drawctx.h> 39 39 #include <surface.h> 40 #include <font/embedded.h> 41 #include <errno.h> 40 42 #include "window.h" 41 43 #include "label.h" … … 58 60 sysarg_t cpt_width; 59 61 sysarg_t cpt_height; 60 font_get_box( &lbl->font, lbl->caption, &cpt_width, &cpt_height);62 font_get_box(lbl->font, lbl->caption, &cpt_width, &cpt_height); 61 63 62 64 if ((widget->width >= cpt_width) && (widget->height >= cpt_height)) { … … 65 67 66 68 drawctx_set_source(&drawctx, &lbl->text); 67 drawctx_set_font(&drawctx, &lbl->font);69 drawctx_set_font(&drawctx, lbl->font); 68 70 69 71 if (lbl->caption) … … 84 86 sysarg_t cpt_width; 85 87 sysarg_t cpt_height; 86 font_get_box( &lbl->font, lbl->caption, &cpt_width, &cpt_height);88 font_get_box(lbl->font, lbl->caption, &cpt_width, &cpt_height); 87 89 88 90 lbl->widget.width_min = cpt_width + 4; … … 99 101 widget_deinit(&lbl->widget); 100 102 free(lbl->caption); 101 font_release( &lbl->font);103 font_release(lbl->font); 102 104 } 103 105 … … 161 163 lbl->caption = str_dup(caption); 162 164 163 font_init(&lbl->font, FONT_DECODER_EMBEDDED, NULL, points); 165 int rc = embedded_font_create(&lbl->font, points); 166 if (rc != EOK) { 167 free(lbl->caption); 168 lbl->caption = NULL; 169 return false; 170 } 164 171 165 172 sysarg_t cpt_width; 166 173 sysarg_t cpt_height; 167 font_get_box( &lbl->font, lbl->caption, &cpt_width, &cpt_height);174 font_get_box(lbl->font, lbl->caption, &cpt_width, &cpt_height); 168 175 169 176 lbl->widget.width_min = cpt_width + 4; -
uspace/lib/gui/label.h
r613d644 r2cc1ec0 51 51 source_t text; 52 52 char *caption; 53 font_t font;53 font_t *font; 54 54 slot_t rewrite; 55 55 } label_t; -
uspace/lib/gui/terminal.c
r613d644 r2cc1ec0 186 186 // for full UTF-32 coverage. 187 187 188 uint16_t glyph = fb_font_glyph(field->ch );188 uint16_t glyph = fb_font_glyph(field->ch, NULL); 189 189 190 190 for (unsigned int y = 0; y < FONT_SCANLINES; y++) { -
uspace/lib/gui/window.c
r613d644 r2cc1ec0 55 55 #include <drawctx.h> 56 56 #include <surface.h> 57 #include <font/embedded.h> 57 58 58 59 #include "common.h" … … 160 161 /* Window caption */ 161 162 162 font_t font; 163 font_init(&font, FONT_DECODER_EMBEDDED, NULL, 16); 164 165 drawctx_set_font(&drawctx, &font); 163 font_t *font; 164 int rc = embedded_font_create(&font, 16); 165 if (rc != EOK) { 166 window_yield(widget->window); 167 return; 168 } 169 170 drawctx_set_font(&drawctx, font); 166 171 source_set_color(&source, widget->window->is_focused ? 167 172 color_caption_focus : color_caption_unfocus); … … 169 174 sysarg_t cpt_width; 170 175 sysarg_t cpt_height; 171 font_get_box( &font, widget->window->caption, &cpt_width, &cpt_height);176 font_get_box(font, widget->window->caption, &cpt_width, &cpt_height); 172 177 173 178 bool draw_title = … … 183 188 } 184 189 185 font_release( &font);190 font_release(font); 186 191 window_yield(widget->window); 187 192 }
Note:
See TracChangeset
for help on using the changeset viewer.