aboutsummaryrefslogtreecommitdiff
path: root/libs/libgui
diff options
context:
space:
mode:
authorMarvin Borner2021-07-01 14:43:53 +0200
committerMarvin Borner2021-07-01 14:43:53 +0200
commit340e841772eb13d9087235b8707c1cfeff8710cb (patch)
tree5f4c3328105a3ab194deef9f05fd05ff3c26b7dc /libs/libgui
parent73a7db5b48eab615a0f6258e0196a1260a88cbbb (diff)
Math improvements and basic plotting
Diffstat (limited to 'libs/libgui')
-rw-r--r--libs/libgui/gfx.c6
-rw-r--r--libs/libgui/gui.c18
-rw-r--r--libs/libgui/gui.h2
-rw-r--r--libs/libgui/widgets.c86
-rw-r--r--libs/libgui/widgets.h22
5 files changed, 130 insertions, 4 deletions
diff --git a/libs/libgui/gfx.c b/libs/libgui/gfx.c
index 913fe59..8f6f163 100644
--- a/libs/libgui/gfx.c
+++ b/libs/libgui/gfx.c
@@ -305,8 +305,8 @@ struct gfx_context *gfx_scale(struct gfx_context *ctx, vec2 size)
y++;
}
- f32 gx = x / (f32)size.x * (ctx->size.x - 1);
- f32 gy = y / (f32)size.y * (ctx->size.y - 1);
+ f64 gx = x / (f64)size.x * (ctx->size.x - 1);
+ f64 gy = y / (f64)size.y * (ctx->size.y - 1);
u32 gxi = (u32)gx;
u32 gyi = (u32)gy;
@@ -317,7 +317,7 @@ struct gfx_context *gfx_scale(struct gfx_context *ctx, vec2 size)
u32 color = 0;
for (u8 i = 0; i < bypp; i++) {
- color |= ((u32)blerpf(GET_COLOR(a, i), GET_COLOR(b, i), GET_COLOR(c, i),
+ color |= ((u32)mblerp(GET_COLOR(a, i), GET_COLOR(b, i), GET_COLOR(c, i),
GET_COLOR(d, i), gx - gxi, gy - gyi))
<< (i << 3);
}
diff --git a/libs/libgui/gui.c b/libs/libgui/gui.c
index 56457e8..2dae302 100644
--- a/libs/libgui/gui.c
+++ b/libs/libgui/gui.c
@@ -486,9 +486,25 @@ vec2 gui_window_size(u32 win_id)
}
/**
- * Window data setters
+ * Widget data getters/setters
*/
+vec2 gui_widget_size(u32 win_id, u32 widget_id)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ gui_error(ENOENT);
+ return widget->bg.size;
+}
+
+struct gfx_context *gui_widget_context(u32 win_id, u32 widget_id, enum gui_layer layer)
+{
+ struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
+ if (!widget)
+ gui_error(ENOENT);
+ return layer == GUI_LAYER_BG ? &widget->bg : &widget->fg;
+}
+
void gui_widget_margin(u32 win_id, u32 widget_id, vec2 margin)
{
struct gui_widget *widget = gui_widget_by_id(win_id, widget_id);
diff --git a/libs/libgui/gui.h b/libs/libgui/gui.h
index 00f49b8..0a4427b 100644
--- a/libs/libgui/gui.h
+++ b/libs/libgui/gui.h
@@ -81,6 +81,8 @@ void gui_popup(const char *text);
*/
vec2 gui_window_size(u32 win_id);
+vec2 gui_widget_size(u32 win_id, u32 widget_id);
+struct gfx_context *gui_widget_context(u32 win_id, u32 widget_id, enum gui_layer layer);
void gui_widget_margin(u32 win_id, u32 widget_id, vec2 margin);
void gui_widget_layout(u32 win_id, u32 widget_id, enum gui_layout layout);
diff --git a/libs/libgui/widgets.c b/libs/libgui/widgets.c
index 1881655..b04f937 100644
--- a/libs/libgui/widgets.c
+++ b/libs/libgui/widgets.c
@@ -1,8 +1,10 @@
// MIT License, Copyright (c) 2021 Marvin Borner
+#include <assert.h>
#include <def.h>
#include <libgui/gui.h>
#include <libgui/widgets.h>
+#include <math.h>
#define TEXT_PAD 2
@@ -26,3 +28,87 @@ void gui_button(u32 window, u32 widget, void (*click)(struct gui_event_mouse *ev
{
gui_button_custom(window, widget, FONT_16, COLOR_WHITE, COLOR_BLACK, click, text);
}
+
+/**
+ * Plot
+ */
+
+static void gui_plot_redraw(struct gui_plot *plot)
+{
+ if (plot->current <= 1)
+ return;
+
+ assert(plot->current <= plot->count);
+
+ gfx_fill(plot->ctx, COLOR_BG);
+
+ f64 min = 0.0, max = 0.0;
+ for (u32 i = 0; i < plot->current; i++) {
+ f64 current = plot->data[i];
+ if (current < min)
+ min = current;
+ if (current > max)
+ max = current;
+ }
+
+ vec2 size = vec2_sub(plot->ctx->size, vec2(2, 2)); // Margin
+ u32 diff = ABS(max - min);
+ u32 step = size.x / (plot->current - 1);
+ f64 scale = (f64)size.y / (diff + 1);
+
+ for (u32 i = 1; i < plot->current; i++) {
+ vec2 prev = vec2((i - 1) * step + 2, (max - plot->data[i - 1]) * scale);
+ vec2 current = vec2(i * step + 2, (max - plot->data[i]) * scale);
+ gfx_draw_line(plot->ctx, prev, current, 1, COLOR_RED);
+ }
+}
+
+void gui_plot_iterate(struct gui_plot *plot)
+{
+ plot->current++;
+ assert(plot->current <= plot->count);
+ gui_plot_redraw(plot);
+}
+
+void gui_plot_destroy(struct gui_plot *plot)
+{
+ free(plot->data);
+ free(plot);
+}
+
+void gui_plot_draw(struct gui_plot *plot)
+{
+ plot->current = plot->count;
+ gui_plot_redraw(plot);
+}
+
+struct gui_plot *gui_plot_data(u32 window, u32 widget, const f64 *data, u32 count)
+{
+ f64 *buffer = malloc(count * sizeof(*data));
+ memcpy(buffer, data, count * sizeof(*data));
+
+ struct gui_plot *plot = malloc(sizeof(*plot));
+ plot->data = buffer;
+ plot->current = 0;
+ plot->count = count;
+ plot->ctx = gui_widget_context(window, widget, GUI_LAYER_FG);
+ gui_fill(window, widget, GUI_LAYER_BG, COLOR_BG);
+ return plot;
+}
+
+struct gui_plot *gui_plot_cubic(u32 window, u32 widget, f64 delta, s32 x_min, s32 x_max, f64 a,
+ f64 b, f64 c, f64 d)
+{
+ assert(x_min < x_max);
+ u32 diff = FABS(x_max - x_min);
+ u32 count = mceil(diff * (1 / delta)) + 1;
+ f64 *data = zalloc(count * sizeof(*data));
+
+ for (u32 i = 0; i < count; i++)
+ data[i] = mcubic(x_min + i * delta, a, b, c, d);
+
+ struct gui_plot *plot = gui_plot_data(window, widget, data, count);
+
+ free(data);
+ return plot;
+}
diff --git a/libs/libgui/widgets.h b/libs/libgui/widgets.h
index 4028af6..64bf6a8 100644
--- a/libs/libgui/widgets.h
+++ b/libs/libgui/widgets.h
@@ -6,9 +6,31 @@
#include <libgui/gfx.h>
#include <libgui/gui.h>
+/**
+ * Button
+ */
+
void gui_button_custom(u32 window, u32 widget, enum font_type font_type, u32 bg, u32 fg,
void (*click)(struct gui_event_mouse *event), const char *text);
void gui_button(u32 window, u32 widget, void (*click)(struct gui_event_mouse *event),
const char *text);
+/**
+ * Plot
+ */
+
+struct gui_plot {
+ f64 *data;
+ u32 current;
+ u32 count;
+ struct gfx_context *ctx;
+};
+
+void gui_plot_iterate(struct gui_plot *plot);
+void gui_plot_destroy(struct gui_plot *plot);
+void gui_plot_draw(struct gui_plot *plot);
+struct gui_plot *gui_plot_data(u32 window, u32 widget, const f64 *data, u32 count);
+struct gui_plot *gui_plot_cubic(u32 window, u32 widget, f64 delta, s32 x_min, s32 x_max, f64 a,
+ f64 b, f64 c, f64 d);
+
#endif