/* * Copyright (c) 2014 Benjamin Otte * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ #include "config.h" #include "graphrenderer.h" #include "graphdata.h" #include "gtksnapshot.h" enum { PROP_0, PROP_DATA, PROP_MINIMUM, PROP_MAXIMUM }; struct _GraphRenderer { GtkWidget parent; GraphData *data; double minimum; double maximum; }; G_DEFINE_TYPE (GraphRenderer, graph_renderer, GTK_TYPE_WIDGET) static void graph_renderer_dispose (GObject *object) { GraphRenderer *self = GRAPH_RENDERER (object); g_clear_object (&self->data); G_OBJECT_CLASS (graph_renderer_parent_class)->dispose (object); } static void graph_renderer_get_property (GObject *object, guint param_id, GValue *value, GParamSpec *pspec) { GraphRenderer *self = GRAPH_RENDERER (object); switch (param_id) { case PROP_DATA: g_value_set_object (value, self->data); break; case PROP_MINIMUM: g_value_set_double (value, self->minimum); break; case PROP_MAXIMUM: g_value_set_double (value, self->maximum); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec); } } static void graph_renderer_set_property (GObject *object, guint param_id, const GValue *value, GParamSpec *pspec) { GraphRenderer *self = GRAPH_RENDERER (object); switch (param_id) { case PROP_DATA: graph_renderer_set_data (self, g_value_get_object (value)); break; case PROP_MINIMUM: if (self->minimum != g_value_get_double (value)) { self->minimum = g_value_get_double (value); g_object_notify_by_pspec (object, pspec); } break; case PROP_MAXIMUM: if (self->maximum != g_value_get_double (value)) { self->maximum = g_value_get_double (value); g_object_notify_by_pspec (object, pspec); } break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec); } } #define MIN_HEIGHT 24 #define MIN_WIDTH (3 * MIN_HEIGHT) static void graph_renderer_measure (GtkWidget *widget, GtkOrientation orientation, int for_size, int *minimum, int *natural, int *minimum_baseline, int *natural_baseline) { if (orientation == GTK_ORIENTATION_HORIZONTAL) *minimum = *natural = MIN_WIDTH; else *minimum = *natural = MIN_HEIGHT; } static void graph_renderer_snapshot (GtkWidget *widget, GtkSnapshot *snapshot) { GraphRenderer *self = GRAPH_RENDERER (widget); double minimum, maximum, diff; double x, y, width, height; cairo_t *cr; GdkRGBA color; guint i, n; #define LINE_WIDTH 1.0 if (self->data == NULL) return; if (self->minimum == -G_MAXDOUBLE) minimum = graph_data_get_minimum (self->data); else minimum = self->minimum; if (self->maximum == G_MAXDOUBLE) maximum = graph_data_get_maximum (self->data); else maximum = self->maximum; diff = maximum - minimum; gtk_widget_get_color (widget, &color); cr = gtk_snapshot_append_cairo (snapshot, &GRAPHENE_RECT_INIT ( 0, 0, gtk_widget_get_width (widget), gtk_widget_get_height (widget) )); cairo_set_line_width (cr, 1.0); x = LINE_WIDTH / 2.0; y = LINE_WIDTH / 2.0; width = gtk_widget_get_width (widget) - LINE_WIDTH; height = gtk_widget_get_height (widget) - LINE_WIDTH; cairo_move_to (cr, x, y + height); if (diff > 0) { n = graph_data_get_n_values (self->data); for (i = 0; i < n; i++) { double val = graph_data_get_value (self->data, i); val = (val - minimum) / diff; val = y + height - val * height; cairo_line_to (cr, x + width * i / (n - 1), val); } } cairo_line_to (cr, x + width, y + height); cairo_close_path (cr); gdk_cairo_set_source_rgba (cr, &color); cairo_stroke_preserve (cr); color.alpha *= 0.2; gdk_cairo_set_source_rgba (cr, &color); cairo_fill (cr); cairo_destroy (cr); } static void graph_renderer_class_init (GraphRendererClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); object_class->dispose = graph_renderer_dispose; object_class->get_property = graph_renderer_get_property; object_class->set_property = graph_renderer_set_property; widget_class->measure = graph_renderer_measure; widget_class->snapshot = graph_renderer_snapshot; g_object_class_install_property (object_class, PROP_DATA, g_param_spec_object ("data", NULL, NULL, graph_data_get_type (), G_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY)); g_object_class_install_property (object_class, PROP_MINIMUM, g_param_spec_double ("minimum", NULL, NULL, -G_MAXDOUBLE, G_MAXDOUBLE, -G_MAXDOUBLE, G_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY)); g_object_class_install_property (object_class, PROP_MINIMUM, g_param_spec_double ("maximum", NULL, NULL, -G_MAXDOUBLE, G_MAXDOUBLE, G_MAXDOUBLE, G_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY)); } static void graph_renderer_init (GraphRenderer *self) { self->minimum = -G_MAXDOUBLE; self->maximum = G_MAXDOUBLE; } GraphRenderer * graph_renderer_new (void) { return g_object_new (graph_renderer_get_type (), NULL); } void graph_renderer_set_data (GraphRenderer *self, GraphData *data) { if (g_set_object (&self->data, data)) g_object_notify (G_OBJECT (self), "data"); gtk_widget_queue_draw (GTK_WIDGET (self)); }