Project

General

Profile

Statistics
| Revision:

root / proj / libs / graph / src / font.c @ 298

History | View | Annotate | Download (6 KB)

1
#include <lcom/lcf.h>
2

    
3
#include "font.h"
4

    
5
#include "xpm_utils.h"
6
#include "graph.h"
7
#include "utils.h"
8
#include "errors.h"
9
#include <assert.h>
10

    
11
struct glyph{
12
    uint16_t w, h;
13
    uint8_t *map;
14
};
15
typedef struct glyph glyph_t;
16
static glyph_t* (glyph_ctor)(const char **xpm){
17
    if(xpm == NULL) return NULL;
18
    glyph_t *ret = malloc(sizeof(glyph_t));
19
    if(ret == NULL) return NULL;
20
    enum xpm_image_type type = XPM_8_8_8_8;
21
    xpm_image_t img;
22
    ret->map = NULL;
23
    uint8_t *map = xpm_load((xpm_map_t)xpm, type, &img);
24
    if(map == NULL){
25
        free(ret);
26
        return NULL;
27
    }
28
    ret->w = img.width;
29
    ret->h = img.height;
30
    ret->map = malloc(ret->w*ret->h*sizeof(uint8_t));
31
    if(ret->map == NULL){
32
        free(ret);
33
        return NULL;
34
    }
35
    for(unsigned i = 0; i < ret->w*ret->h; ++i){
36
        ret->map[i] = map[4*i+3];
37
    }
38
    return ret;
39
}
40
static void (glyph_dtor)(glyph_t *p){
41
    if(p == NULL) return;
42
    free(p->map);
43
    free(p);
44
}
45
static int (glyph_draw_to_alpha_buffer)(const glyph_t *p, int16_t x, int16_t y, uint8_t *alp_buf, uint16_t W, uint16_t H){
46
    if(p == NULL) return NULL_PTR;
47
    for(int16_t h = 0; h < p->h; ++h){
48
        for(int16_t w = 0; w < p->w; ++w){
49
            uint8_t a = *(p->map + w + h*p->w);
50
            int16_t x_ = x+w, y_ = y-p->h+h;
51
            unsigned pos = x_ +y_*W;
52
            alp_buf[pos] = a;
53
        }
54
    }
55
    return SUCCESS;
56
}
57

    
58
struct font{
59
    size_t nchars;
60
    glyph_t **glyphs;
61
};
62

    
63
font_t* (font_ctor)(const char *s){
64
    font_t *ret = malloc(sizeof(font_t));
65
    if(ret == NULL) return NULL;
66
    ret->nchars = 128;
67
    ret->glyphs = malloc(ret->nchars*sizeof(glyph_t*));
68
    if(ret->glyphs == NULL){
69
        free(ret->glyphs);
70
        free(ret);
71
        return NULL;
72
    }
73
    int good = false;
74
    char filepath[1024];
75
    for(size_t i = 0; i < ret->nchars; ++i){
76
        sprintf(filepath, "%s/ascii%03d.xpm2", s, i);
77
        char **xpm = xpm_load_xpm2(filepath);
78
        ret->glyphs[i] = glyph_ctor((const char**)xpm);
79
        if(ret->glyphs[i] != NULL) good = true;
80
    }
81
    if(good) return ret;
82
    else{
83
        //font_dtor(ret);
84
        return NULL;
85
    }
86
}
87
void (font_dtor)(font_t *p){
88
    if(p == NULL) return;
89
    for(size_t i = 0; i < p->nchars; ++i)
90
        glyph_dtor(p->glyphs[i]);
91
    free(p->glyphs);
92
    free(p);
93
}
94

    
95
struct text{
96
    const font_t *fnt;
97
    char *txt;
98
    int16_t x, y;
99
    int size;
100
    uint32_t color;
101
    enum text_valign valign;
102
    enum text_halign halign;
103
};
104
text_t* (text_ctor)(const font_t *fnt, const char *txt){
105
    if(fnt == NULL) return NULL;
106
    text_t *ret = malloc(sizeof(text_t));
107
    if(ret == NULL) return NULL;
108
    ret->fnt = fnt;
109
    ret->txt = NULL;
110
    text_set_text(ret, txt);
111
    ret->x = 0;
112
    ret->y = 0;
113
    ret->size = 25;
114
    ret->color = GRAPH_BLACK;
115
    ret->valign = text_valign_top;
116
    ret->halign = text_halign_left;
117
    return ret;
118
}
119
void (text_dtor)(text_t *p){
120
    if(p == NULL) return;
121
    free(p->txt);
122
    free(p);
123
}
124
void (text_set_text) (text_t *p, const char *txt){
125
    size_t sz = strlen(txt);
126
    p->txt = realloc(p->txt, (sz+1)*sizeof(char));
127
    if(p->txt == NULL) return;
128
    strcpy(p->txt, txt);
129
}
130
char* (text_get_string)(const text_t *p){return p->txt; }
131
void (text_set_pos)   (text_t *p, int16_t x, int16_t y   ){ p->x = x; p->y = y; }
132
void (text_set_size)  (text_t *p, unsigned size          ){ p->size = size    ; }
133
void (text_set_color) (text_t *p, uint32_t color         ){ p->color = color  ; }
134
void (text_set_valign)(text_t *p, enum text_valign valign){ p->valign = valign; }
135
void (text_set_halign)(text_t *p, enum text_halign halign){ p->halign = halign; }
136
int16_t (text_get_x)  (const text_t *p){ return p->x; }
137
int16_t (text_get_y)  (const text_t *p){ return p->y; }
138

    
139
int (text_draw)(const text_t *p){
140
    if(p == NULL) return NULL_PTR;
141
    int ret = SUCCESS;
142
    // Get buffer with rescaled text
143
    uint8_t *alp_new_buf = NULL;
144
    uint16_t newH, newW;{
145
        const size_t len = strlen(p->txt);
146
        uint16_t W = 0, H = 0; {
147
            for(size_t i = 0; i < len; ++i){
148
                const glyph_t *g = p->fnt->glyphs[(size_t)p->txt[i]];
149
                if(g != NULL){ W += g->w; H = max(H, g->h); }
150
            }
151
        }
152
        uint8_t *alp_buf = malloc(W*H);
153
        if(alp_buf == NULL) return ALLOC_ERROR;{
154
            int16_t y = H;
155
            int16_t x = 0;
156
            for(size_t i = 0; i < len; ++i){
157
                const glyph_t *g = p->fnt->glyphs[(size_t)p->txt[i]];
158
                if(g != NULL){
159
                    if((ret = glyph_draw_to_alpha_buffer(g, x, y, alp_buf, W, H))) return ret;
160
                    x += g->w;
161
                }
162
            }
163
        }
164

    
165
        double factor = (double)p->size/(double)H;
166

    
167
        newH = H*factor;
168
        newW = W*factor;
169
        alp_new_buf = malloc(newW*newH);
170
        if(alp_new_buf == NULL) return ALLOC_ERROR;
171

    
172
        for(size_t newy = 0; newy < newH; ++newy){
173
            size_t y = newy/factor;
174
            for(size_t newx = 0; newx < newW; ++newx){
175
                size_t x = newx/factor;
176
                *(alp_new_buf+newx+newy*newW) = *(alp_buf+x+y*W);
177
            }
178
        }
179
        free(alp_buf);
180
    }
181
    // Get initial value of x
182
    int16_t initx;{
183
        switch(p->halign){
184
            case text_halign_left  : initx = p->x         ; break;
185
            case text_halign_center: initx = p->x - newW/2; break;
186
            case text_halign_right : initx = p->x - newW  ; break;
187
            default: return LOGIC_ERROR;
188
        }
189
    }
190
    // Get initial value of y
191
    int16_t inity;{
192
        switch(p->valign){
193
            case text_valign_top   : inity = p->y         ; break;
194
            case text_valign_center: inity = p->y - newH/2; break;
195
            case text_valign_bottom: inity = p->y - newH  ; break;
196
            default: return LOGIC_ERROR;
197
        }
198
    }
199
    // Draw text
200
    for(int16_t newy = 0; newy < newH; ++newy){
201
        for(int16_t newx = 0; newx < newW; ++newx){
202
            uint8_t a = *(alp_new_buf+newx+newy*newW);
203
            if(a < 0x7F) graph_set_pixel(initx+newx,inity+newy,p->color);
204
        }
205
    }
206
    free(alp_new_buf);
207
    return SUCCESS;
208
}