]> git.scottworley.com Git - batteryviewer/blobdiff - chart.c
Don't draw connecting lines across large gaps
[batteryviewer] / chart.c
diff --git a/chart.c b/chart.c
index 8b05d33e0ecd28f02abc8419fa54f76f2b7254ba..1fa76c542656f611eb2974fb1323f7d174cd8090 100644 (file)
--- a/chart.c
+++ b/chart.c
@@ -4,6 +4,7 @@
  */
 
 #include "chart.h"
+#include <float.h>
 #include <gtk/gtk.h>
 
 #define BV_CHART_PRIVATE(obj)                                                  \
@@ -13,6 +14,7 @@ typedef struct _BVChartPrivate BVChartPrivate;
 
 struct _BVChartPrivate {
   GArray *points;
+  float minx, miny, maxx, maxy;
 };
 
 struct BVChartPoint {
@@ -22,7 +24,46 @@ struct BVChartPoint {
 G_DEFINE_TYPE_WITH_CODE(BVChart, bv_chart, GTK_TYPE_DRAWING_AREA,
                         G_ADD_PRIVATE(BVChart))
 
-static void bv_chart_class_init(BVChartClass *klass __attribute__((unused))) {}
+typedef void (*cairo_draw_func)(cairo_t *cr, double x, double y);
+
+static gboolean bv_chart_draw(GtkWidget *widget, cairo_t *cr) {
+  BVChart *chart = BV_CHART(widget);
+  BVChartPrivate *priv = bv_chart_get_instance_private(chart);
+
+  cairo_set_source_rgb(cr, 1.0, 1.0, 1.0);
+  cairo_paint(cr);
+
+  if (priv->points->len < 2)
+    return TRUE;
+
+  GtkAllocation allocation;
+  gtk_widget_get_allocation(widget, &allocation);
+
+  int margin = 3;
+  float xscale = (allocation.width - 2 * margin) / (priv->maxx - priv->minx);
+  float yscale = (allocation.height - 2 * margin) / (priv->miny - priv->maxy);
+  cairo_set_source_rgb(cr, 0.0, 0.0, 1.0);
+  float gap_threshold = 3.0;
+  float prevx = 0.0;
+  for (guint i = 0; i < priv->points->len; i++) {
+    struct BVChartPoint *p =
+        &g_array_index(priv->points, struct BVChartPoint, i);
+    gboolean is_gap = p->x - prevx > gap_threshold;
+    cairo_draw_func f = is_gap ? cairo_move_to : cairo_line_to;
+    float screen_x = xscale * (p->x - priv->minx) + margin;
+    float screen_y = yscale * (p->y - priv->maxy) + margin;
+    f(cr, screen_x, screen_y);
+    prevx = p->x;
+  }
+  cairo_stroke(cr);
+
+  return TRUE;
+}
+
+static void bv_chart_class_init(BVChartClass *klass) {
+  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(klass);
+  widget_class->draw = bv_chart_draw;
+}
 
 static void bv_chart_init(BVChart *chart) {
   gtk_widget_set_has_window(GTK_WIDGET(chart), FALSE);
@@ -31,6 +72,10 @@ static void bv_chart_init(BVChart *chart) {
   gboolean clear_ = FALSE;
   priv->points =
       g_array_new(zero_terminated, clear_, sizeof(struct BVChartPoint));
+  priv->minx = FLT_MAX;
+  priv->miny = FLT_MAX;
+  priv->maxx = FLT_MIN;
+  priv->maxy = FLT_MIN;
 }
 
 GtkWidget *bv_chart_new() {
@@ -39,6 +84,14 @@ GtkWidget *bv_chart_new() {
 
 void bv_chart_add_point(BVChart *chart, float x, float y) {
   BVChartPrivate *priv = bv_chart_get_instance_private(chart);
+  if (x < priv->minx)
+    priv->minx = x;
+  if (y < priv->miny)
+    priv->miny = y;
+  if (x > priv->maxx)
+    priv->maxx = x;
+  if (y > priv->maxy)
+    priv->maxy = y;
   struct BVChartPoint p = {.x = x, .y = y};
   g_array_append_val(priv->points, p);
 }