mirror of https://github.com/Gnucash/gnucash
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
760 lines
21 KiB
760 lines
21 KiB
/* vim: set sw=8: -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
|
/*
|
|
* gog-chart.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., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
* USA
|
|
*/
|
|
|
|
#include <goffice/goffice-config.h>
|
|
#include <goffice/graph/gog-chart-impl.h>
|
|
#include <goffice/graph/gog-plot-impl.h>
|
|
#include <goffice/graph/gog-graph-impl.h>
|
|
#include <goffice/graph/gog-style.h>
|
|
#include <goffice/graph/gog-view.h>
|
|
#include <goffice/graph/gog-axis.h>
|
|
#include <goffice/graph/gog-grid.h>
|
|
#include <goffice/graph/gog-grid-line.h>
|
|
#include <goffice/graph/gog-renderer.h>
|
|
|
|
#include <gsf/gsf-impl-utils.h>
|
|
#include <glib/gi18n.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
enum {
|
|
CHART_PROP_0,
|
|
CHART_PROP_CARDINALITY_VALID
|
|
};
|
|
|
|
static GType gog_chart_view_get_type (void);
|
|
static GObjectClass *chart_parent_klass;
|
|
|
|
static void
|
|
gog_chart_update (GogObject *obj)
|
|
{
|
|
GogChart *chart = GOG_CHART (obj);
|
|
unsigned full = chart->full_cardinality;
|
|
unsigned visible = chart->visible_cardinality;
|
|
|
|
gog_chart_get_cardinality (chart, NULL, NULL);
|
|
|
|
if (full != chart->full_cardinality ||
|
|
visible != chart->visible_cardinality)
|
|
g_object_notify (G_OBJECT (chart), "cardinality-valid");
|
|
}
|
|
|
|
static void
|
|
gog_chart_finalize (GObject *obj)
|
|
{
|
|
GogChart *chart = GOG_CHART (obj);
|
|
|
|
/* on exit the role remove routines are not called */
|
|
g_slist_free (chart->plots);
|
|
g_slist_free (chart->axes);
|
|
|
|
(chart_parent_klass->finalize) (obj);
|
|
}
|
|
|
|
static void
|
|
gog_chart_get_property (GObject *obj, guint param_id,
|
|
GValue *value, GParamSpec *pspec)
|
|
{
|
|
GogChart *chart = GOG_CHART (obj);
|
|
switch (param_id) {
|
|
case CHART_PROP_CARDINALITY_VALID:
|
|
g_value_set_boolean (value, chart->cardinality_valid);
|
|
break;
|
|
|
|
default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, param_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gog_chart_children_reordered (GogObject *obj)
|
|
{
|
|
GSList *ptr, *accum = NULL;
|
|
GogChart *chart = GOG_CHART (obj);
|
|
|
|
for (ptr = obj->children; ptr != NULL ; ptr = ptr->next)
|
|
if (IS_GOG_PLOT (ptr->data))
|
|
accum = g_slist_prepend (accum, ptr->data);
|
|
g_slist_free (chart->plots);
|
|
chart->plots = g_slist_reverse (accum);
|
|
|
|
gog_chart_request_cardinality_update (chart);
|
|
}
|
|
|
|
static void
|
|
role_plot_post_add (GogObject *parent, GogObject *plot)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
gboolean ok = TRUE;
|
|
|
|
/* APPEND to keep order, there won't be that many */
|
|
chart->plots = g_slist_append (chart->plots, plot);
|
|
gog_chart_request_cardinality_update (chart);
|
|
|
|
if (chart->plots->next == NULL)
|
|
ok = gog_chart_axis_set_assign (chart,
|
|
gog_plot_axis_set_pref (GOG_PLOT (plot)));
|
|
ok |= gog_plot_axis_set_assign (GOG_PLOT (plot),
|
|
chart->axis_set);
|
|
|
|
/* a quick post condition to keep us on our toes */
|
|
g_return_if_fail (ok);
|
|
}
|
|
|
|
static void
|
|
role_plot_pre_remove (GogObject *parent, GogObject *plot)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
gog_plot_axis_clear (GOG_PLOT (plot), GOG_AXIS_SET_ALL);
|
|
chart->plots = g_slist_remove (chart->plots, plot);
|
|
gog_chart_request_cardinality_update (chart);
|
|
}
|
|
|
|
static gboolean
|
|
role_grid_can_add (GogObject const *parent)
|
|
{
|
|
GogChart const *chart = GOG_CHART (parent);
|
|
return chart->grid == NULL && chart->axis_set == GOG_AXIS_SET_XY;
|
|
}
|
|
static void
|
|
role_grid_post_add (GogObject *parent, GogObject *child)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
g_return_if_fail (chart->grid == NULL);
|
|
chart->grid = child;
|
|
}
|
|
|
|
static void
|
|
role_grid_pre_remove (GogObject *parent, GogObject *grid)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
g_return_if_fail (chart->grid == grid);
|
|
chart->grid = NULL;
|
|
}
|
|
|
|
static gboolean
|
|
axis_can_add (GogObject const *parent, GogAxisType t)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
if (chart->axis_set == GOG_AXIS_SET_UNKNOWN)
|
|
return FALSE;
|
|
return (chart->axis_set & (1 << t)) != 0;
|
|
}
|
|
static gboolean
|
|
axis_can_remove (GogObject const *child)
|
|
{
|
|
return NULL == gog_axis_contributors (GOG_AXIS (child));
|
|
}
|
|
|
|
static void
|
|
axis_post_add (GogObject *axis, GogAxisType t)
|
|
{
|
|
GogChart *chart = GOG_CHART (axis->parent);
|
|
g_object_set (G_OBJECT (axis), "type", (int)t, NULL);
|
|
chart->axes = g_slist_prepend (chart->axes, axis);
|
|
}
|
|
|
|
static void
|
|
axis_pre_remove (GogObject *parent, GogObject *axis)
|
|
{
|
|
GogChart *chart = GOG_CHART (parent);
|
|
gog_axis_clear_contributors (GOG_AXIS (axis));
|
|
chart->axes = g_slist_remove (chart->axes, axis);
|
|
}
|
|
|
|
static gboolean x_axis_can_add (GogObject const *parent) { return axis_can_add (parent, GOG_AXIS_X); }
|
|
static void x_axis_post_add (GogObject *parent, GogObject *child) { axis_post_add (child, GOG_AXIS_X); }
|
|
static gboolean y_axis_can_add (GogObject const *parent) { return axis_can_add (parent, GOG_AXIS_Y); }
|
|
static void y_axis_post_add (GogObject *parent, GogObject *child) { axis_post_add (child, GOG_AXIS_Y); }
|
|
static gboolean z_axis_can_add (GogObject const *parent) { return axis_can_add (parent, GOG_AXIS_Z); }
|
|
static void z_axis_post_add (GogObject *parent, GogObject *child) { axis_post_add (child, GOG_AXIS_Z); }
|
|
static gboolean circular_axis_can_add (GogObject const *parent) { return axis_can_add (parent, GOG_AXIS_CIRCULAR); }
|
|
static void circular_axis_post_add (GogObject *parent, GogObject *child) { axis_post_add (child, GOG_AXIS_CIRCULAR); }
|
|
static gboolean radial_axis_can_add (GogObject const *parent) { return axis_can_add (parent, GOG_AXIS_RADIAL); }
|
|
static void radial_axis_post_add (GogObject *parent, GogObject *child) { axis_post_add (child, GOG_AXIS_RADIAL); }
|
|
|
|
static GogObjectRole const roles[] = {
|
|
{ N_("Legend"), "GogLegend", 0,
|
|
GOG_POSITION_COMPASS, GOG_POSITION_E|GOG_POSITION_ALIGN_CENTER, GOG_OBJECT_NAME_BY_ROLE,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, { -1 } },
|
|
{ N_("Title"), "GogLabel", 1,
|
|
GOG_POSITION_COMPASS, GOG_POSITION_N|GOG_POSITION_ALIGN_CENTER, GOG_OBJECT_NAME_BY_ROLE,
|
|
NULL, NULL, NULL, NULL, NULL, NULL, { -1 } },
|
|
{ N_("Grid"), "GogGrid", 0,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
role_grid_can_add, NULL, NULL, role_grid_post_add, role_grid_pre_remove, NULL, { -1 } },
|
|
{ N_("X-Axis"), "GogAxis", 1,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
x_axis_can_add, axis_can_remove, NULL, x_axis_post_add, axis_pre_remove, NULL,
|
|
{ GOG_AXIS_X } },
|
|
{ N_("Y-Axis"), "GogAxis", 2,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
y_axis_can_add, axis_can_remove, NULL, y_axis_post_add, axis_pre_remove, NULL,
|
|
{ GOG_AXIS_Y } },
|
|
{ N_("Z-Axis"), "GogAxis", 3,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
z_axis_can_add, axis_can_remove, NULL, z_axis_post_add, axis_pre_remove, NULL,
|
|
{ GOG_AXIS_Z } },
|
|
{ N_("Circular-Axis"), "GogAxis", 1,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
circular_axis_can_add, axis_can_remove, NULL, circular_axis_post_add, axis_pre_remove, NULL,
|
|
{ GOG_AXIS_CIRCULAR } },
|
|
{ N_("Radial-Axis"), "GogAxis", 2,
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_ROLE,
|
|
radial_axis_can_add, axis_can_remove, NULL, radial_axis_post_add, axis_pre_remove, NULL,
|
|
{ GOG_AXIS_RADIAL } },
|
|
{ N_("Plot"), "GogPlot", 4, /* keep the axis before the plots */
|
|
GOG_POSITION_SPECIAL, GOG_POSITION_SPECIAL, GOG_OBJECT_NAME_BY_TYPE,
|
|
NULL, NULL, NULL, role_plot_post_add, role_plot_pre_remove, NULL, { -1 } }
|
|
};
|
|
|
|
static void
|
|
gog_chart_class_init (GogObjectClass *gog_klass)
|
|
{
|
|
GObjectClass *gobject_klass = (GObjectClass *)gog_klass;
|
|
|
|
chart_parent_klass = g_type_class_peek_parent (gog_klass);
|
|
gobject_klass->finalize = gog_chart_finalize;
|
|
gobject_klass->get_property = gog_chart_get_property;
|
|
|
|
g_object_class_install_property (gobject_klass, CHART_PROP_CARDINALITY_VALID,
|
|
g_param_spec_boolean ("cardinality-valid", "cardinality-valid",
|
|
"Is the charts cardinality currently vaid",
|
|
FALSE, G_PARAM_READABLE));
|
|
|
|
gog_klass->view_type = gog_chart_view_get_type ();
|
|
gog_klass->update = gog_chart_update;
|
|
gog_klass->children_reordered = gog_chart_children_reordered;
|
|
gog_object_register_roles (gog_klass, roles, G_N_ELEMENTS (roles));
|
|
}
|
|
|
|
static void
|
|
gog_chart_init (GogChart *chart)
|
|
{
|
|
chart->x = 0;
|
|
chart->y = 0;
|
|
chart->cols = 0;
|
|
chart->rows = 0;
|
|
|
|
/* start as true so that we can queue an update when it changes */
|
|
chart->cardinality_valid = TRUE;
|
|
chart->axis_set = GOG_AXIS_SET_UNKNOWN;
|
|
}
|
|
|
|
GSF_CLASS (GogChart, gog_chart,
|
|
gog_chart_class_init, gog_chart_init,
|
|
GOG_OUTLINED_OBJECT_TYPE)
|
|
|
|
/**
|
|
* gog_chart_get_position :
|
|
* @chart : const #GogChart
|
|
* @x :
|
|
* @y :
|
|
* @cols :
|
|
* @rows :
|
|
*
|
|
* Returns TRUE if the chart has been positioned.
|
|
**/
|
|
gboolean
|
|
gog_chart_get_position (GogChart const *chart,
|
|
unsigned *x, unsigned *y, unsigned *cols, unsigned *rows)
|
|
{
|
|
g_return_val_if_fail (GOG_CHART (chart), FALSE);
|
|
|
|
if (chart->cols <= 0 || chart->rows <= 0)
|
|
return FALSE;
|
|
|
|
if (x != NULL) *x = chart->x;
|
|
if (y != NULL) *y = chart->y;
|
|
if (cols != NULL) *cols = chart->cols;
|
|
if (rows != NULL) *rows = chart->rows;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* gog_chart_set_position :
|
|
* @chart : #GogChart
|
|
* @x :
|
|
* @y :
|
|
* @cols :
|
|
* @rows :
|
|
*
|
|
**/
|
|
void
|
|
gog_chart_set_position (GogChart *chart,
|
|
unsigned x, unsigned y, unsigned cols, unsigned rows)
|
|
{
|
|
g_return_if_fail (GOG_CHART (chart) != NULL);
|
|
|
|
if (chart->x == x && chart->y == y &&
|
|
chart->cols == cols && chart->rows == rows)
|
|
return;
|
|
|
|
chart->x = x;
|
|
chart->y = y;
|
|
chart->cols = cols;
|
|
chart->rows = rows;
|
|
|
|
gog_graph_validate_chart_layout (GOG_GRAPH (GOG_OBJECT (chart)->parent));
|
|
gog_object_emit_changed (GOG_OBJECT (chart), TRUE);
|
|
}
|
|
|
|
void
|
|
gog_chart_get_cardinality (GogChart *chart, unsigned *full, unsigned *visible)
|
|
{
|
|
GSList *ptr;
|
|
unsigned tmp_full, tmp_visible;
|
|
|
|
g_return_if_fail (GOG_CHART (chart) != NULL);
|
|
|
|
if (!chart->cardinality_valid) {
|
|
chart->cardinality_valid = TRUE;
|
|
chart->full_cardinality = chart->visible_cardinality = 0;
|
|
for (ptr = chart->plots ; ptr != NULL ; ptr = ptr->next) {
|
|
gog_plot_get_cardinality (ptr->data, &tmp_full, &tmp_visible);
|
|
chart->full_cardinality += tmp_full;
|
|
chart->visible_cardinality += tmp_visible;
|
|
}
|
|
}
|
|
|
|
if (full != NULL)
|
|
*full = chart->full_cardinality;
|
|
if (visible != NULL)
|
|
*visible = chart->visible_cardinality;
|
|
}
|
|
|
|
void
|
|
gog_chart_request_cardinality_update (GogChart *chart)
|
|
{
|
|
g_return_if_fail (GOG_CHART (chart) != NULL);
|
|
|
|
if (chart->cardinality_valid) {
|
|
chart->cardinality_valid = FALSE;
|
|
gog_object_request_update (GOG_OBJECT (chart));
|
|
}
|
|
}
|
|
|
|
void
|
|
gog_chart_foreach_elem (GogChart *chart, gboolean only_visible,
|
|
GogEnumFunc handler, gpointer data)
|
|
{
|
|
GSList *ptr;
|
|
|
|
g_return_if_fail (GOG_CHART (chart) != NULL);
|
|
g_return_if_fail (chart->cardinality_valid);
|
|
|
|
for (ptr = chart->plots ; ptr != NULL ; ptr = ptr->next)
|
|
gog_plot_foreach_elem (ptr->data, only_visible, handler, data);
|
|
}
|
|
|
|
GSList *
|
|
gog_chart_get_plots (GogChart const *chart)
|
|
{
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, NULL);
|
|
return chart->plots;
|
|
}
|
|
|
|
GogAxisSet
|
|
gog_chart_axis_set (GogChart const *chart)
|
|
{
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, GOG_AXIS_SET_UNKNOWN);
|
|
return chart->axis_set;
|
|
}
|
|
|
|
gboolean
|
|
gog_chart_axis_set_is_valid (GogChart const *chart, GogAxisSet type)
|
|
{
|
|
GSList *ptr;
|
|
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, FALSE);
|
|
|
|
for (ptr = chart->plots ; ptr != NULL ; ptr = ptr->next)
|
|
if (!gog_plot_axis_set_is_valid (ptr->data, type))
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gog_chart_add_axis (GogChart *chart, GogAxisType type)
|
|
{
|
|
unsigned i = G_N_ELEMENTS (roles);
|
|
while (i-- > 0)
|
|
if (roles[i].user.i == (int)type) {
|
|
gog_object_add_by_role (GOG_OBJECT (chart), roles + i, NULL);
|
|
return;
|
|
}
|
|
g_warning ("unknown axis type %d", type);
|
|
}
|
|
|
|
gboolean
|
|
gog_chart_axis_set_assign (GogChart *chart, GogAxisSet axis_set)
|
|
{
|
|
GogAxis *axis;
|
|
GSList *ptr;
|
|
GogAxisType type;
|
|
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, FALSE);
|
|
|
|
if (chart->axis_set == axis_set)
|
|
return TRUE;
|
|
chart->axis_set = axis_set;
|
|
|
|
if (chart->grid != NULL && axis_set != GOG_AXIS_SET_XY) {
|
|
GogObject *grid = chart->grid; /* clear_parent clears ::grid */
|
|
gog_object_clear_parent (GOG_OBJECT (grid));
|
|
g_object_unref (grid);
|
|
} else if (chart->grid == NULL && axis_set == GOG_AXIS_SET_XY)
|
|
gog_object_add_by_name (GOG_OBJECT (chart), "Grid", NULL);
|
|
|
|
/* Add at least 1 instance of any required axis */
|
|
for (type = 0 ; type < GOG_AXIS_TYPES ; type++)
|
|
if ((axis_set & (1 << type))) {
|
|
GSList *tmp = gog_chart_get_axis (chart, type);
|
|
if (tmp == NULL)
|
|
gog_chart_add_axis (chart, type);
|
|
else
|
|
g_slist_free (tmp);
|
|
}
|
|
|
|
/* link the plots */
|
|
for (ptr = chart->plots ; ptr != NULL ; ptr = ptr->next)
|
|
if (!gog_plot_axis_set_assign (ptr->data, axis_set))
|
|
return FALSE;
|
|
|
|
/* remove any existing axis that do not fit this scheme */
|
|
for (ptr = GOG_OBJECT (chart)->children ; ptr != NULL ; ) {
|
|
axis = ptr->data;
|
|
ptr = ptr->next; /* list may change under us */
|
|
if (IS_GOG_AXIS (axis)) {
|
|
type = -1;
|
|
g_object_get (G_OBJECT (axis), "type", &type, NULL);
|
|
if (type < 0 || type >= GOG_AXIS_TYPES) {
|
|
g_warning ("Invalid axis");
|
|
continue;
|
|
}
|
|
|
|
if (0 == (axis_set & (1 << type))) {
|
|
gog_object_clear_parent (GOG_OBJECT (axis));
|
|
g_object_unref (axis);
|
|
}
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* gog_chart_get_axis :
|
|
* @chart : #GogChart
|
|
* @target : #GogAxisType
|
|
*
|
|
* Return a list which the caller must free of all axis of type @target
|
|
* associated with @chart.
|
|
**/
|
|
GSList *
|
|
gog_chart_get_axis (GogChart const *chart, GogAxisType target)
|
|
{
|
|
GSList *ptr, *res = NULL;
|
|
GogAxis *axis;
|
|
int type;
|
|
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, NULL);
|
|
|
|
for (ptr = GOG_OBJECT (chart)->children ; ptr != NULL ; ptr = ptr->next) {
|
|
axis = ptr->data;
|
|
if (IS_GOG_AXIS (axis)) {
|
|
type = -1;
|
|
g_object_get (G_OBJECT (axis), "type", &type, NULL);
|
|
if (type < 0 || type >= GOG_AXIS_TYPES) {
|
|
g_warning ("Invalid axis");
|
|
continue;
|
|
}
|
|
if (type == target)
|
|
res = g_slist_prepend (res, axis);
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* gog_chart_get_grid :
|
|
* @chart : #GogChart
|
|
*
|
|
* Returns the grid associated with @chart if one exists
|
|
* otherwise NULL.
|
|
**/
|
|
GogGrid *
|
|
gog_chart_get_grid (GogChart const *chart)
|
|
{
|
|
g_return_val_if_fail (GOG_CHART (chart) != NULL, NULL);
|
|
return GOG_GRID (chart->grid);
|
|
}
|
|
|
|
/*********************************************************************/
|
|
|
|
typedef struct {
|
|
GogOutlinedView base;
|
|
|
|
GogViewAllocation plot_area;
|
|
} GogChartView;
|
|
typedef GogOutlinedViewClass GogChartViewClass;
|
|
|
|
#define GOG_CHART_VIEW_TYPE (gog_chart_view_get_type ())
|
|
#define GOG_CHART_VIEW(o) (G_TYPE_CHECK_INSTANCE_CAST ((o), GOG_CHART_VIEW_TYPE, GogChartView))
|
|
#define IS_GOG_CHART_VIEW(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), GOG_CHART_VIEW_TYPE))
|
|
|
|
static GogViewClass *cview_parent_klass;
|
|
|
|
GogViewAllocation const *
|
|
gog_chart_view_get_plot_area (GogView const *view)
|
|
{
|
|
g_return_val_if_fail ((GOG_CHART_VIEW (view) != NULL), NULL);
|
|
|
|
return & (GOG_CHART_VIEW(view)->plot_area);
|
|
}
|
|
|
|
static void
|
|
child_request (GogView *view, GogViewAllocation *res,
|
|
GogViewAllocation const *plot_area,
|
|
gboolean allocate)
|
|
{
|
|
GSList *ptr;
|
|
GogView *child;
|
|
GogAxis const *axis;
|
|
GogViewRequisition req;
|
|
GogViewAllocation allocation;
|
|
|
|
for (ptr = view->children; ptr != NULL ; ptr = ptr->next) {
|
|
child = ptr->data;
|
|
if (child->model->position != GOG_POSITION_SPECIAL ||
|
|
!IS_GOG_AXIS (child->model))
|
|
continue;
|
|
|
|
axis = GOG_AXIS (child->model);
|
|
req.w = req.h = 0.;
|
|
gog_view_size_request (child, &req);
|
|
allocation = *plot_area;
|
|
switch (gog_axis_get_atype (axis)) {
|
|
case GOG_AXIS_X:
|
|
if (req.h > 0) {
|
|
res->h -= req.h;
|
|
allocation.h = req.h;
|
|
if (gog_axis_get_pos (axis) == GOG_AXIS_AT_HIGH) {
|
|
allocation.y = res->y;
|
|
res->y += req.h;
|
|
} else
|
|
allocation.y = res->y + res->h;
|
|
}
|
|
|
|
|
|
break;
|
|
case GOG_AXIS_Y:
|
|
if (req.w > 0) {
|
|
res->w -= req.w;
|
|
allocation.w = req.w;
|
|
if (gog_axis_get_pos (axis) == GOG_AXIS_AT_LOW) {
|
|
allocation.x = res->x;
|
|
res->x += req.w;
|
|
} else
|
|
allocation.x = res->x + res->w;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (allocate)
|
|
gog_view_size_allocate (child, &allocation);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gog_chart_view_size_allocate (GogView *view, GogViewAllocation const *allocation)
|
|
{
|
|
GSList *ptr;
|
|
GogView *child;
|
|
GogChart *chart = GOG_CHART (view->model);
|
|
GogViewAllocation res = *allocation;
|
|
GogViewAllocation tmp, axis_alloc;
|
|
|
|
(cview_parent_klass->size_allocate) (view, &res);
|
|
|
|
res = view->residual;
|
|
switch (chart->axis_set) {
|
|
|
|
case GOG_AXIS_SET_XY:
|
|
{
|
|
GogViewPadding axis_padding, padding = {0., 0., 0., 0.};
|
|
|
|
tmp = res;
|
|
child_request (view, &res, &res, FALSE);
|
|
|
|
/* FIXME: we need to iterate until convergence */
|
|
for (ptr = view->children; ptr != NULL ; ptr = ptr->next) {
|
|
child = ptr->data;
|
|
if (child->model->position != GOG_POSITION_SPECIAL ||
|
|
!IS_GOG_AXIS (child->model))
|
|
continue;
|
|
|
|
gog_axis_view_padding_request (child, &axis_padding, &res);
|
|
padding.wr = MAX (padding.wr, axis_padding.wr);
|
|
padding.wl = MAX (padding.wl, axis_padding.wl);
|
|
padding.hb = MAX (padding.hb, axis_padding.hb);
|
|
padding.ht = MAX (padding.ht, axis_padding.ht);
|
|
}
|
|
res.x += padding.wl;
|
|
res.w -= padding.wl + padding.wr;
|
|
res.y += padding.ht;
|
|
res.h -= padding.ht + padding.hb;
|
|
|
|
for (ptr = view->children; ptr != NULL ; ptr = ptr->next) {
|
|
child = ptr->data;
|
|
if (child->model->position != GOG_POSITION_SPECIAL ||
|
|
!IS_GOG_AXIS (child->model))
|
|
continue;
|
|
|
|
switch (gog_axis_get_atype (GOG_AXIS (child->model)))
|
|
{
|
|
case GOG_AXIS_X:
|
|
axis_alloc = tmp;
|
|
axis_alloc.x = res.x;
|
|
axis_alloc.w = res.w;
|
|
gog_view_size_allocate (child, &axis_alloc);
|
|
break;
|
|
|
|
case GOG_AXIS_Y:
|
|
axis_alloc = tmp;
|
|
axis_alloc.y = res.y;
|
|
axis_alloc.h = res.h;
|
|
gog_view_size_allocate (child, &axis_alloc);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case GOG_AXIS_SET_RADAR:
|
|
/* Give the axes the whole residual area. */
|
|
for (ptr = view->children; ptr != NULL ; ptr = ptr->next) {
|
|
child = ptr->data;
|
|
if (IS_GOG_AXIS (child->model))
|
|
gog_view_size_allocate (child, &res);
|
|
}
|
|
break;
|
|
case GOG_AXIS_SET_NONE:
|
|
break;
|
|
|
|
case GOG_AXIS_SET_UNKNOWN:
|
|
return;
|
|
default:
|
|
g_warning ("only have layout engine for xy, radar, and none currently");
|
|
return;
|
|
}
|
|
|
|
/* overlay all the plots in the residual */
|
|
for (ptr = view->children; ptr != NULL ; ptr = ptr->next) {
|
|
child = ptr->data;
|
|
if (child->model->position == GOG_POSITION_SPECIAL &&
|
|
(IS_GOG_PLOT (child->model) || child->model == chart->grid))
|
|
gog_view_size_allocate (child, &res);
|
|
}
|
|
|
|
GOG_CHART_VIEW(view)->plot_area = res;
|
|
}
|
|
|
|
static void
|
|
gog_chart_view_init (GogChartView *cview)
|
|
{
|
|
}
|
|
|
|
static void
|
|
grid_line_render (GSList *start_ptr, GogViewAllocation const *bbox)
|
|
{
|
|
GSList *ptr, *child_ptr;
|
|
GogView *child_view, *axis_child_view;
|
|
|
|
/* Render minor lines first */
|
|
for (ptr = start_ptr; ptr != NULL; ptr = ptr->next) {
|
|
child_view = ptr->data;
|
|
if (IS_GOG_AXIS (child_view->model)) {
|
|
for (child_ptr = child_view->children; child_ptr != NULL; child_ptr = child_ptr->next) {
|
|
axis_child_view = child_ptr->data;
|
|
if (IS_GOG_GRID_LINE (axis_child_view->model) &&
|
|
gog_grid_line_is_minor (GOG_GRID_LINE (axis_child_view->model)))
|
|
gog_view_render (axis_child_view, bbox);
|
|
}
|
|
}
|
|
}
|
|
/* then render major lines */
|
|
for (ptr = start_ptr; ptr != NULL; ptr = ptr->next) {
|
|
child_view = ptr->data;
|
|
if (IS_GOG_AXIS (child_view->model)) {
|
|
for (child_ptr = child_view->children; child_ptr != NULL; child_ptr = child_ptr->next) {
|
|
axis_child_view = child_ptr->data;
|
|
if (IS_GOG_GRID_LINE (axis_child_view->model) &&
|
|
!gog_grid_line_is_minor (GOG_GRID_LINE (axis_child_view->model)))
|
|
gog_view_render (axis_child_view, bbox);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
gog_chart_view_render (GogView *view, GogViewAllocation const *bbox)
|
|
{
|
|
GSList *ptr;
|
|
GogView *child_view;
|
|
gboolean grid_line_rendered = FALSE;
|
|
|
|
cview_parent_klass->render (view, bbox);
|
|
|
|
/* KLUDGE: render grid lines before axis */
|
|
for (ptr = view->children ; ptr != NULL ; ptr = ptr->next) {
|
|
child_view = ptr->data;
|
|
if (!grid_line_rendered && IS_GOG_AXIS (child_view->model)) {
|
|
grid_line_render (ptr, bbox);
|
|
grid_line_rendered = TRUE;
|
|
}
|
|
gog_view_render (ptr->data, bbox);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gog_chart_view_class_init (GogChartViewClass *gview_klass)
|
|
{
|
|
GogViewClass *view_klass = (GogViewClass *) gview_klass;
|
|
GogOutlinedViewClass *oview_klass = (GogOutlinedViewClass *) gview_klass;
|
|
|
|
cview_parent_klass = g_type_class_peek_parent (gview_klass);
|
|
view_klass->size_allocate = gog_chart_view_size_allocate;
|
|
view_klass->clip = TRUE;
|
|
view_klass->render = gog_chart_view_render;
|
|
oview_klass->call_parent_render = FALSE;
|
|
}
|
|
|
|
static GSF_CLASS (GogChartView, gog_chart_view,
|
|
gog_chart_view_class_init, gog_chart_view_init,
|
|
GOG_OUTLINED_VIEW_TYPE)
|