/* vim: set sw=8: -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * go-barcol.c * * Copyright (C) 2003-2004 Jody Goldberg (jody@gnome.org) * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 * USA */ #include #include "gog-barcol.h" #include #include #include #include #include #include #include #include #include #include #include enum { BARCOL_PROP_0, BARCOL_PROP_GAP_PERCENTAGE, BARCOL_PROP_OVERLAP_PERCENTAGE, BARCOL_PROP_HORIZONTAL }; static GogObjectClass *gog_barcol_parent_klass; static GType gog_barcol_view_get_type (void); static void gog_barcol_plot_set_property (GObject *obj, guint param_id, GValue const *value, GParamSpec *pspec) { GogBarColPlot *barcol = GOG_BARCOL_PLOT (obj); switch (param_id) { case BARCOL_PROP_GAP_PERCENTAGE: barcol->gap_percentage = g_value_get_int (value); break; case BARCOL_PROP_OVERLAP_PERCENTAGE: barcol->overlap_percentage = g_value_get_int (value); break; case BARCOL_PROP_HORIZONTAL: barcol->horizontal = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, param_id, pspec); return; /* NOTE : RETURN */ } gog_object_emit_changed (GOG_OBJECT (obj), TRUE); } static void gog_barcol_plot_get_property (GObject *obj, guint param_id, GValue *value, GParamSpec *pspec) { GogBarColPlot *barcol = GOG_BARCOL_PLOT (obj); switch (param_id) { case BARCOL_PROP_GAP_PERCENTAGE: g_value_set_int (value, barcol->gap_percentage); break; case BARCOL_PROP_OVERLAP_PERCENTAGE: g_value_set_int (value, barcol->overlap_percentage); break; case BARCOL_PROP_HORIZONTAL: g_value_set_boolean (value, barcol->horizontal); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, param_id, pspec); break; } } static char const * gog_barcol_plot_type_name (G_GNUC_UNUSED GogObject const *item) { /* xgettext : the base for how to name bar/col plot objects * eg The 2nd bar/col plot in a chart will be called * PlotBarCol2 */ return N_("PlotBarCol"); } extern gpointer gog_barcol_plot_pref (GogBarColPlot *barcol, GOCmdContext *cc); static void gog_barcol_plot_populate_editor (GogObject *item, GogEditor *editor, G_GNUC_UNUSED GogDataAllocator *dalloc, GOCmdContext *cc) { gog_editor_add_page (editor, gog_barcol_plot_pref (GOG_BARCOL_PLOT (item), cc), _("Properties")); (GOG_OBJECT_CLASS(gog_barcol_parent_klass)->populate_editor) (item, editor, dalloc, cc); } static gboolean gog_barcol_swap_x_and_y (GogPlot1_5d *model) { return GOG_BARCOL_PLOT (model)->horizontal; } static void gog_barcol_update_stacked_and_percentage (GogPlot1_5d *model, double **vals, GogErrorBar **errors, unsigned const *lengths) { unsigned i, j; double neg_sum, pos_sum, tmp, errplus, errminus, tmpmin, tmpmax; for (i = model->num_elements ; i-- > 0 ; ) { neg_sum = pos_sum = 0.; tmpmin = DBL_MAX; tmpmax = -DBL_MAX; for (j = 0 ; j < model->num_series ; j++) { if (i >= lengths[j]) continue; tmp = vals[j][i]; if (!go_finite (tmp)) continue; if (gog_error_bar_is_visible (errors[j])) { gog_error_bar_get_bounds (errors[j], i, &errminus, &errplus); errminus = errminus > 0. ? errminus : 0.; errplus = errplus > 0. ? errplus : 0.; } else errplus = errminus = 0.; if (tmp > 0.) { pos_sum += tmp; errminus = (pos_sum - errminus < neg_sum)? neg_sum - pos_sum + errminus: 0.; } else { neg_sum += tmp; errplus = (neg_sum + errplus > pos_sum)? neg_sum - pos_sum + errplus: 0.; } if (tmpmin > neg_sum - errminus) tmpmin = neg_sum - errminus; if (tmpmax < pos_sum + errplus) tmpmax = pos_sum + errplus; } if (GOG_1_5D_STACKED == model->type) { if (model->minima > tmpmin) model->minima = tmpmin; if (model->maxima < tmpmax) model->maxima = tmpmax; } else { if (model->minima > tmpmin / (pos_sum - neg_sum)) model->minima = tmpmin / (pos_sum - neg_sum); if (model->maxima < tmpmax / (pos_sum - neg_sum)) model->maxima = tmpmax / (pos_sum - neg_sum); } } } static GOData * gog_barcol_axis_get_bounds (GogPlot *plot, GogAxisType axis, GogPlotBoundInfo *bounds) { GogPlot1_5d *model = GOG_PLOT1_5D (plot); GogPlot1_5dClass *plot1_5d_klass = GOG_PLOT1_5D_CLASS (gog_barcol_parent_klass); GOData *data; data = (plot1_5d_klass->base.axis_get_bounds) (plot, axis, bounds); if (axis == gog_axis_get_atype (gog_plot1_5d_get_index_axis (model))) { bounds->val.minima -= .5; bounds->val.maxima += .5; bounds->logical.minima = -.5; bounds->center_on_ticks = FALSE; } return data; } static void gog_barcol_plot_class_init (GogPlot1_5dClass *gog_plot_1_5d_klass) { GObjectClass *gobject_klass = (GObjectClass *) gog_plot_1_5d_klass; GogObjectClass *gog_object_klass = (GogObjectClass *) gog_plot_1_5d_klass; GogPlotClass *plot_klass = (GogPlotClass *) gog_plot_1_5d_klass; gog_barcol_parent_klass = g_type_class_peek_parent (gog_plot_1_5d_klass); gobject_klass->set_property = gog_barcol_plot_set_property; gobject_klass->get_property = gog_barcol_plot_get_property; g_object_class_install_property (gobject_klass, BARCOL_PROP_GAP_PERCENTAGE, g_param_spec_int ("gap-percentage", "gap percentage", "The padding around each group as a percentage of their width", 0, 500, 150, G_PARAM_READWRITE | GOG_PARAM_PERSISTENT)); g_object_class_install_property (gobject_klass, BARCOL_PROP_OVERLAP_PERCENTAGE, g_param_spec_int ("overlap-percentage", "overlap percentage", "The distance between series as a percentage of their width", -100, 100, 0, G_PARAM_READWRITE | GOG_PARAM_PERSISTENT)); g_object_class_install_property (gobject_klass, BARCOL_PROP_HORIZONTAL, g_param_spec_boolean ("horizontal", "horizontal", "horizontal bars or vertical columns", FALSE, G_PARAM_READWRITE | GOG_PARAM_PERSISTENT)); gog_object_klass->type_name = gog_barcol_plot_type_name; gog_object_klass->populate_editor = gog_barcol_plot_populate_editor; gog_object_klass->view_type = gog_barcol_view_get_type (); plot_klass->desc.series.style_fields = GOG_STYLE_OUTLINE | GOG_STYLE_FILL; plot_klass->axis_get_bounds = gog_barcol_axis_get_bounds; gog_plot_1_5d_klass->swap_x_and_y = gog_barcol_swap_x_and_y; gog_plot_1_5d_klass->update_stacked_and_percentage = gog_barcol_update_stacked_and_percentage; } static void gog_barcol_plot_init (GogBarColPlot *model) { model->gap_percentage = 150; GOG_PLOT1_5D (model)->support_series_lines = TRUE; } GSF_DYNAMIC_CLASS (GogBarColPlot, gog_barcol_plot, gog_barcol_plot_class_init, gog_barcol_plot_init, GOG_PLOT1_5D_TYPE) /*****************************************************************************/ typedef GogPlotView GogBarColView; typedef GogPlotViewClass GogBarColViewClass; /** * FIXME FIXME FIXME Wrong description * barcol_draw_rect : * @rend : #GogRenderer * @flip : * @base : #GogViewAllocation * @rect : #GogViewAllocation * * A utility routine to build a vpath in @rect. @rect is assumed to be in * coordinates relative to @base with 0,0 as the upper left. @flip transposes * @rect and rotates it to put the origin in the bottom left. Play fast and * loose with coordinates to keep widths >= 1. If we allow things to be less * the background bleeds through. **/ static void barcol_draw_rect (GogRenderer *rend, gboolean flip, GogAxisMap *x_map, GogAxisMap *y_map, GogViewAllocation const *rect) { ArtVpath path[6]; double x0, x1, y0, y1; if (flip) { x0 = gog_axis_map_to_view (x_map, rect->y); x1 = gog_axis_map_to_view (x_map, rect->y + rect->h); y0 = gog_axis_map_to_view (y_map, rect->x); y1 = gog_axis_map_to_view (y_map, rect->x + rect->w); } else { x0 = gog_axis_map_to_view (x_map, rect->x); x1 = gog_axis_map_to_view (x_map, rect->x + rect->w); y0 = gog_axis_map_to_view (y_map, rect->y); y1 = gog_axis_map_to_view (y_map, rect->y + rect->h); } path[0].x = path[3].x = path[4].x = x0; path[1].x = path[2].x = x1; path[0].y = path[1].y = path[4].y = y0; path[2].y = path[3].y = y1; path[0].code = ART_MOVETO; path[1].code = ART_LINETO; path[2].code = ART_LINETO; path[3].code = ART_LINETO; path[4].code = ART_LINETO; path[5].code = ART_END; gog_renderer_draw_sharp_polygon (rend, path, fabs (x1 - x0) < 3. || fabs (y1 - y0) < 3.); } typedef struct { double plus; double minus; double x; double y; } ErrorBarData; static void gog_barcol_view_render (GogView *view, GogViewAllocation const *bbox) { GogBarColPlot const *model = GOG_BARCOL_PLOT (view->model); GogPlot1_5d const *gog_1_5d_model = GOG_PLOT1_5D (view->model); GogSeries1_5d const *series; GogChart *chart = GOG_CHART (view->model->parent); GogChartMap *chart_map; GogViewAllocation work; GogViewAllocation const *area; GogRenderer *rend = view->renderer; GogAxisMap *x_map, *y_map, *map; gboolean is_vertical = ! (model->horizontal); double **vals, sum, neg_base, pos_base, tmp; double x; double col_step, group_step, offset, data_scale; unsigned i, j; unsigned num_elements = gog_1_5d_model->num_elements; unsigned num_series = gog_1_5d_model->num_series; GogPlot1_5dType const type = gog_1_5d_model->type; GogStyle **styles; ErrorBarData **error_data; GogErrorBar **errors; GogSeriesLines **lines; ArtVpath **paths; GSList *ptr; unsigned *lengths; double plus, minus; GogObjectRole const *role = NULL; if (num_elements <= 0 || num_series <= 0) return; area = gog_chart_view_get_plot_area (view->parent); chart_map = gog_chart_map_new (chart, area, GOG_PLOT (model)->axis[GOG_AXIS_X], GOG_PLOT (model)->axis[GOG_AXIS_Y], NULL, FALSE); if (!gog_chart_map_is_valid (chart_map)) { gog_chart_map_free (chart_map); return; } x_map = gog_chart_map_get_axis_map (chart_map, 0); y_map = gog_chart_map_get_axis_map (chart_map, 1); map = is_vertical ? y_map : x_map; vals = g_alloca (num_series * sizeof (double *)); lengths = g_alloca (num_series * sizeof (unsigned)); styles = g_alloca (num_series * sizeof (GogStyle *)); errors = g_alloca (num_series * sizeof (GogErrorBar *)); error_data = g_alloca (num_series * sizeof (ErrorBarData *)); lines = g_alloca (num_series * sizeof (GogSeriesLines *)); paths = g_alloca (num_series * sizeof (ArtVpath *)); i = 0; for (ptr = gog_1_5d_model->base.series ; ptr != NULL ; ptr = ptr->next) { series = ptr->data; if (!gog_series_is_valid (GOG_SERIES (series))) continue; vals[i] = go_data_vector_get_values ( GO_DATA_VECTOR (series->base.values[1].data)); lengths[i] = go_data_vector_get_len ( GO_DATA_VECTOR (series->base.values[1].data)); styles[i] = GOG_STYLED_OBJECT (series)->style; errors[i] = series->errors; if (gog_error_bar_is_visible (series->errors)) error_data[i] = g_malloc (sizeof (ErrorBarData) * lengths[i]); else error_data[i] = NULL; if (series->has_series_lines) { if (!role) role = gog_object_find_role_by_name ( GOG_OBJECT (series), "Series lines"); lines[i] = GOG_SERIES_LINES ( gog_object_get_child_by_role (GOG_OBJECT (series), role)); paths[i] = g_new (ArtVpath, lengths[i] * 2 - 1); for (j = 0; j < lengths[i] - 1; j++) { paths[i][j * 2].code = ART_MOVETO; paths[i][j * 2 + 1].code = ART_LINETO; } paths[i][j * 2].code = ART_END; } else lines[i] = NULL; i++; } /* work in coordinates drawing bars from the top */ col_step = 1. - model->overlap_percentage / 100.; group_step = model->gap_percentage / 100.; work.h = 1.0 / (1. + ((num_series - 1.0) * col_step) + group_step); col_step *= work.h; offset = (col_step * (num_series - 1.0) + work.h) / 2.0; data_scale = 1.0; for (i = 0 ; i < num_elements ; i++) { if (type == GOG_1_5D_AS_PERCENTAGE) { sum = 0.; for (j = num_series ; j-- > 0 ; ) { if (i >= lengths[j]) continue; tmp = vals[j][i]; if (!gog_axis_map_finite (map, tmp)) continue; if (tmp > 0.) sum += tmp; else sum -= tmp; } data_scale = (fabs (go_sub_epsilon (sum)) > 0.0) ? 1.0 / sum : 1.0; } pos_base = neg_base = 0.0; for (j = 0 ; j < num_series ; j++) { work.y = (double) j * col_step + (double) i - offset + 1.0; if (i >= lengths[j]) continue; tmp = vals[j][i]; if (!gog_axis_map_finite (map, tmp)) continue; if (gog_error_bar_is_visible (errors[j])) { gog_error_bar_get_bounds (errors[j], i, &minus, &plus); } tmp *= data_scale; if (tmp >= 0.) { work.x = pos_base; work.w = tmp; if (GOG_1_5D_NORMAL != type) pos_base += tmp; } else { work.x = neg_base + tmp; work.w = -tmp; if (GOG_1_5D_NORMAL != type) neg_base += tmp; } gog_renderer_push_style (view->renderer, styles[j]); barcol_draw_rect (rend, is_vertical, x_map, y_map, &work); gog_renderer_pop_style (view->renderer); if (gog_error_bar_is_visible (errors[j])) { x = tmp > 0 ? work.x + work.w: work.x; error_data[j][i].plus = plus * data_scale; error_data[j][i].minus = minus * data_scale; if (is_vertical) { error_data[j][i].x = work.y + work.h / 2.0; error_data[j][i].y = x; } else { error_data[j][i].x = x; error_data[j][i].y = work.y + work.h / 2.0; } } if (lines[j] != NULL) { x = tmp > 0 ? work.x + work.w: work.x; if (is_vertical) { if (i > 0) { paths[j][i * 2 - 1].x = gog_axis_map_to_view (x_map, work.y); paths[j][i * 2 - 1].y = gog_axis_map_to_view (y_map, x); } paths[j][i * 2].x = gog_axis_map_to_view (x_map, work.y + work.h); paths[j][i * 2].y = gog_axis_map_to_view (y_map, x); } else { if (i > 0) { paths[j][i * 2 - 1].x = gog_axis_map_to_view (x_map, x); paths[j][i * 2 - 1].y = gog_axis_map_to_view (y_map, work.y); } paths[j][i * 2].x = gog_axis_map_to_view (x_map, x); paths[j][i * 2].y = gog_axis_map_to_view (y_map, work.y + work.h); } } } } /*Now draw error bars and clean*/ for (i = 0; i < num_series; i++) if (gog_error_bar_is_visible (errors[i])) { for (j = 0; j < lengths[i]; j++) gog_error_bar_render (errors[i], view->renderer, x_map, y_map, error_data[i][j].x , error_data[i][j].y, error_data[i][j].minus, error_data[i][j].plus, model->horizontal); g_free (error_data[i]); } /* Draw series lines if any */ for (i = 0; i < num_series; i++) if (lines[i] != NULL) { gog_series_lines_render (lines[i], view->renderer, bbox, paths[i], FALSE); g_free (paths[i]); } gog_chart_map_free (chart_map); } static gboolean gog_barcol_view_info_at_point (GogView *view, double x, double y, GogObject const *cur_selection, GogObject **obj, char **name) { if (obj != NULL) *obj = view->model; if (name != NULL) *name = g_strdup (gog_object_get_name (GOG_OBJECT (view->model))); return TRUE; } static void gog_barcol_view_class_init (GogViewClass *view_klass) { view_klass->render = gog_barcol_view_render; view_klass->info_at_point = gog_barcol_view_info_at_point; view_klass->clip = TRUE; } GSF_DYNAMIC_CLASS (GogBarColView, gog_barcol_view, gog_barcol_view_class_init, NULL, GOG_PLOT_VIEW_TYPE)