1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
|
// MIT License, Copyright (c) 2020 Marvin Borner
// Some GUI functions
#include <assert.h>
#include <bmp.h>
#include <gui.h>
#include <mem.h>
#include <psf.h>
#include <str.h>
#include <sys.h>
#include <vesa.h>
struct font *font;
void gui_write_char(struct vbe *vbe, int x, int y, const u32 c[3], char ch)
{
int bypp = vbe->bpp >> 3;
int pos = x * bypp + y * vbe->pitch;
char *draw = (char *)&vbe->fb[pos];
u32 stride = font->char_size / font->height;
for (int cy = 0; cy < font->height; cy++) {
for (int cx = 0; cx < font->width; cx++) {
u8 bits = font->chars[ch * font->char_size + cy * stride + cx / 8];
u8 bit = bits >> (7 - cx % 8) & 1;
if (bit) {
draw[bypp * cx] = c[2];
draw[bypp * cx + 1] = c[1];
draw[bypp * cx + 2] = c[0];
}
}
draw += vbe->pitch;
}
}
void gui_write(struct vbe *vbe, int x, int y, const u32 c[3], char *text)
{
for (u32 i = 0; i < strlen(text); i++) {
gui_write_char(vbe, x + i * font->width, y, c, text[i]);
}
}
void gui_load_image(struct window *win, char *path, int x, int y)
{
struct bmp *bmp = bmp_load(path);
assert(bmp && bmp->width + x <= win->width);
assert(bmp && bmp->height + y <= win->height);
// TODO: Support padding with odd widths
u8 *srcfb = bmp->data;
u8 *destfb = win->fb;
int bypp = bmp->bpp >> 3;
for (u32 cy = 0; cy < bmp->height; cy++) {
for (u32 cx = 0; cx < bmp->width; cx++) {
destfb[bypp * cx + 0] = srcfb[bypp * cx + 0];
destfb[bypp * cx + 1] = srcfb[bypp * cx + 1];
destfb[bypp * cx + 2] = srcfb[bypp * cx + 2];
}
srcfb += bmp->pitch;
destfb += win->pitch;
}
}
void gui_load_wallpaper(struct window *win, char *path)
{
gui_load_image(win, path, 0, 0);
}
// TODO: Optimize!
void gui_win_on_win(struct window *dest, struct window *src, int x, int y)
{
// Optimization?
/* if (src->width == dest->width && src->height == dest->height && src->x == 0 && */
/* dest->y == 0) { */
/* memcpy(dest->fb, src->fb, dest->pitch * dest->height); */
/* return; */
/* } */
int bypp = dest->bpp >> 3;
u8 *srcfb = src->fb;
u8 *destfb = &dest->fb[x * bypp + y * dest->pitch];
for (u32 cy = 0; cy < src->height - 1; cy++) {
for (u32 cx = 0; cx < src->width - 1; cx++) {
destfb[bypp * cx + 0] = srcfb[bypp * cx + 0];
destfb[bypp * cx + 1] = srcfb[bypp * cx + 1];
destfb[bypp * cx + 2] = srcfb[bypp * cx + 2];
}
srcfb += src->pitch;
destfb += dest->pitch;
}
}
void gui_draw_rectangle(struct window *win, int x1, int y1, int x2, int y2, const u32 color[3])
{
int bypp = win->bpp >> 3;
int pos1 = x1 * bypp + y1 * win->pitch;
u8 *draw = &win->fb[pos1];
for (int i = 0; i < y2 - y1; i++) {
for (int j = 0; j < x2 - x1; j++) {
draw[bypp * j] = color[2];
draw[bypp * j + 1] = color[1];
draw[bypp * j + 2] = color[0];
}
draw += win->pitch;
}
}
void gui_fill(struct window *win, const u32 color[3])
{
gui_draw_rectangle(win, 0, 0, win->width - 1, win->height - 1, color);
}
void gui_border(struct window *win, const u32 color[3], u32 width)
{
if (width <= 0)
return;
int bypp = win->bpp >> 3;
u8 *draw = win->fb;
for (u32 i = 0; i < win->height; i++) {
for (u32 j = 0; j < win->width; j++) {
if (j <= width - 1 || i <= width - 1 ||
j - win->width + width + 1 <= width ||
i - win->height + width + 1 <= width) {
draw[bypp * j + 0] = color[2];
draw[bypp * j + 1] = color[1];
draw[bypp * j + 2] = color[0];
}
}
draw += win->pitch;
}
}
void gui_init(char *font_path)
{
font = psf_parse(read(font_path));
}
|