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.
gnucash/lib/goffice/graph/go-data-simple.c

538 lines
14 KiB

/* vim: set sw=8: -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* go-data-simple.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/go-data-simple.h>
#include <goffice/graph/go-data-impl.h>
#include <goffice/utils/go-math.h>
#include <gsf/gsf-impl-utils.h>
#include <glib/gi18n.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
struct _GODataScalarVal {
GODataScalar base;
double val;
char *str;
};
typedef GODataScalarClass GODataScalarValClass;
static GObjectClass *scalar_val_parent_klass;
static void
go_data_scalar_val_finalize (GObject *obj)
{
GODataScalarVal *val = (GODataScalarVal *)obj;
if (val->str != NULL) {
g_free (val->str);
val->str = NULL;
}
(*scalar_val_parent_klass->finalize) (obj);
}
static GOData *
go_data_scalar_val_dup (GOData const *src)
{
GODataScalarVal *dst = g_object_new (G_OBJECT_TYPE (src), NULL);
GODataScalarVal const *src_val = (GODataScalarVal const *)src;
dst->val = src_val->val;
return GO_DATA (dst);
}
static gboolean
go_data_scalar_val_eq (GOData const *a, GOData const *b)
{
GODataScalarVal const *sval_a = (GODataScalarVal const *)a;
GODataScalarVal const *sval_b = (GODataScalarVal const *)b;
/* GOData::eq is used for identity, not arithmetic */
return sval_a->val == sval_b->val;
}
static char *
go_data_scalar_val_as_str (GOData const *dat)
{
return g_strdup (go_data_scalar_get_str (GO_DATA_SCALAR (dat)));
}
static gboolean
go_data_scalar_val_from_str (GOData *dat, char const *str)
{
GODataScalarVal *sval = (GODataScalarVal *)dat;
double tmp;
char *end;
errno = 0; /* strto(ld) sets errno, but does not clear it. */
tmp = strtod (str, &end);
if (end == str || *end != '\0' || errno == ERANGE)
return FALSE;
g_free (sval->str);
sval->str = NULL;
sval->val = tmp;
return TRUE;
}
static double
go_data_scalar_val_get_value (GODataScalar *dat)
{
GODataScalarVal const *sval = (GODataScalarVal const *)dat;
return sval->val;
}
static char const *
go_data_scalar_val_get_str (GODataScalar *dat)
{
GODataScalarVal *sval = (GODataScalarVal *)dat;
if (sval->str == NULL)
sval->str = g_strdup_printf ("%g", sval->val);
return sval->str;
}
static void
go_data_scalar_val_class_init (GObjectClass *gobject_klass)
{
GODataClass *godata_klass = (GODataClass *) gobject_klass;
GODataScalarClass *scalar_klass = (GODataScalarClass *) gobject_klass;
scalar_val_parent_klass = g_type_class_peek_parent (gobject_klass);
gobject_klass->finalize = go_data_scalar_val_finalize;
godata_klass->dup = go_data_scalar_val_dup;
godata_klass->eq = go_data_scalar_val_eq;
godata_klass->as_str = go_data_scalar_val_as_str;
godata_klass->from_str = go_data_scalar_val_from_str;
scalar_klass->get_value = go_data_scalar_val_get_value;
scalar_klass->get_str = go_data_scalar_val_get_str;
}
GSF_CLASS (GODataScalarVal, go_data_scalar_val,
go_data_scalar_val_class_init, NULL,
GO_DATA_SCALAR_TYPE)
GOData *
go_data_scalar_val_new (double val)
{
GODataScalarVal *res = g_object_new (GO_DATA_SCALAR_VAL_TYPE, NULL);
res->val = val;
return GO_DATA (res);
}
/*****************************************************************************/
struct _GODataScalarStr {
GODataScalar base;
char const *str;
gboolean needs_free;
};
typedef GODataScalarClass GODataScalarStrClass;
static GObjectClass *scalar_str_parent_klass;
static void
go_data_scalar_str_finalize (GObject *obj)
{
GODataScalarStr *str = (GODataScalarStr *)obj;
if (str->needs_free && str->str != NULL) {
g_free ((char *)str->str);
str->str = NULL;
}
(*scalar_str_parent_klass->finalize) (obj);
}
static GOData *
go_data_scalar_str_dup (GOData const *src)
{
GODataScalarStr *dst = g_object_new (G_OBJECT_TYPE (src), NULL);
GODataScalarStr const *src_val = (GODataScalarStr const *)src;
dst->needs_free = TRUE;
dst->str = g_strdup (src_val->str);
return GO_DATA (dst);
}
static gboolean
go_data_scalar_str_eq (GOData const *a, GOData const *b)
{
GODataScalarStr const *str_a = (GODataScalarStr const *)a;
GODataScalarStr const *str_b = (GODataScalarStr const *)b;
return 0 == strcmp (str_a->str, str_b->str);
}
static char *
go_data_scalar_str_as_str (GOData const *dat)
{
GODataScalarStr const *str = (GODataScalarStr const *)dat;
return g_strdup (str->str);
}
static gboolean
go_data_scalar_str_from_str (GOData *dat, char const *string)
{
GODataScalarStr *str = (GODataScalarStr *)dat;
if (str->str == string)
return TRUE;
if (str->needs_free)
g_free ((char *)str->str);
str->str = g_strdup (string);
str->needs_free = TRUE;
return TRUE;
}
static double
go_data_scalar_str_get_value (GODataScalar *dat)
{
return go_nan;
}
static char const *
go_data_scalar_str_get_str (GODataScalar *dat)
{
GODataScalarStr const *str = (GODataScalarStr const *)dat;
return str->str;
}
static void
go_data_scalar_str_class_init (GObjectClass *gobject_klass)
{
GODataClass *godata_klass = (GODataClass *) gobject_klass;
GODataScalarClass *scalar_klass = (GODataScalarClass *) gobject_klass;
scalar_str_parent_klass = g_type_class_peek_parent (gobject_klass);
gobject_klass->finalize = go_data_scalar_str_finalize;
godata_klass->dup = go_data_scalar_str_dup;
godata_klass->eq = go_data_scalar_str_eq;
godata_klass->as_str = go_data_scalar_str_as_str;
godata_klass->from_str = go_data_scalar_str_from_str;
scalar_klass->get_value = go_data_scalar_str_get_value;
scalar_klass->get_str = go_data_scalar_str_get_str;
}
static void
go_data_scalar_str_init (GObject *obj)
{
GODataScalarStr *str = (GODataScalarStr *)obj;
str->str = "";
str->needs_free = FALSE;
}
GSF_CLASS (GODataScalarStr, go_data_scalar_str,
go_data_scalar_str_class_init, go_data_scalar_str_init,
GO_DATA_SCALAR_TYPE)
GOData *
go_data_scalar_str_new (char const *str, gboolean needs_free)
{
GODataScalarStr *res = g_object_new (GO_DATA_SCALAR_STR_TYPE, NULL);
res->str = str;
res->needs_free = needs_free;
return GO_DATA (res);
}
void
go_data_scalar_str_set_str (GODataScalarStr *str,
char const *text, gboolean needs_free)
{
if (str->str == text)
return;
if (str->needs_free)
g_free ((char *)str->str);
str->str = text;
str->needs_free = needs_free;
go_data_emit_changed (GO_DATA (str));
}
/*****************************************************************************/
struct _GODataVectorVal {
GODataVector base;
unsigned n;
double const *val;
};
typedef GODataVectorClass GODataVectorValClass;
static GObjectClass *vector_val_parent_klass;
static void
go_data_vector_val_finalize (GObject *obj)
{
/* GODataVectorVal *val = (GODataVectorVal *)obj; */
(*vector_val_parent_klass->finalize) (obj);
}
static GOData *
go_data_vector_val_dup (GOData const *src)
{
GODataVectorVal *dst = g_object_new (G_OBJECT_TYPE (src), NULL);
GODataVectorVal const *src_val = (GODataVectorVal const *)src;
dst->val = src_val->val;
dst->n = src_val->n;
return GO_DATA (dst);
}
static gboolean
go_data_vector_val_eq (GOData const *a, GOData const *b)
{
GODataVectorVal const *val_a = (GODataVectorVal const *)a;
GODataVectorVal const *val_b = (GODataVectorVal const *)b;
/* GOData::eq is used for identity, not arithmetic */
return val_a->val == val_b->val && val_a->n == val_b->n;
}
static void
go_data_vector_val_load_len (GODataVector *vec)
{
vec->base.flags |= GO_DATA_VECTOR_LEN_CACHED;
vec->len = ((GODataVectorVal *)vec)->n;
}
static void
go_data_vector_val_load_values (GODataVector *vec)
{
GODataVectorVal const *val = (GODataVectorVal const *)vec;
double minimum = DBL_MAX, maximum = -DBL_MAX;
int i = val->n;
vec->values = (double *)val->val;
while (i-- > 0) {
if (minimum > val->val[i])
minimum = val->val[i];
if (maximum < val->val[i])
maximum = val->val[i];
}
vec->minimum = minimum;
vec->maximum = maximum;
vec->base.flags |= GO_DATA_CACHE_IS_VALID;
}
static double
go_data_vector_val_get_value (GODataVector *vec, unsigned i)
{
return vec->values[i];
}
static char *
go_data_vector_val_get_str (GODataVector *vec, unsigned i)
{
return g_strdup_printf ("%g", vec->values[i]);
}
static void
go_data_vector_val_class_init (GObjectClass *gobject_klass)
{
GODataClass *godata_klass = (GODataClass *) gobject_klass;
GODataVectorClass *vector_klass = (GODataVectorClass *) gobject_klass;
vector_val_parent_klass = g_type_class_peek_parent (gobject_klass);
gobject_klass->finalize = go_data_vector_val_finalize;
godata_klass->dup = go_data_vector_val_dup;
godata_klass->eq = go_data_vector_val_eq;
godata_klass->as_str = NULL;
godata_klass->from_str = NULL;
vector_klass->load_len = go_data_vector_val_load_len;
vector_klass->load_values = go_data_vector_val_load_values;
vector_klass->get_value = go_data_vector_val_get_value;
vector_klass->get_str = go_data_vector_val_get_str;
}
GSF_CLASS (GODataVectorVal, go_data_vector_val,
go_data_vector_val_class_init, NULL,
GO_DATA_VECTOR_TYPE)
GOData *
go_data_vector_val_new (double const *val, unsigned n)
{
GODataVectorVal *res = g_object_new (GO_DATA_VECTOR_VAL_TYPE, NULL);
res->val = val;
res->n = n;
return GO_DATA (res);
}
/*****************************************************************************/
struct _GODataVectorStr {
GODataVector base;
char const * const *str;
unsigned n;
GOTranslateFunc translate_func;
gpointer translate_data;
GDestroyNotify translate_notify;
};
typedef GODataVectorClass GODataVectorStrClass;
static GObjectClass *vector_str_parent_klass;
static void
go_data_vector_str_finalize (GObject *obj)
{
/* GODataVectorStr *str = (GODataVectorStr *)obj; */
(*vector_str_parent_klass->finalize) (obj);
}
static GOData *
go_data_vector_str_dup (GOData const *src)
{
GODataVectorStr *dst = g_object_new (G_OBJECT_TYPE (src), NULL);
GODataVectorStr const *src_val = (GODataVectorStr const *)src;
dst->n = src_val->n;
dst->str = src_val->str;
return GO_DATA (dst);
}
static gboolean
go_data_vector_str_eq (GOData const *a, GOData const *b)
{
GODataVectorStr const *str_a = (GODataVectorStr const *)a;
GODataVectorStr const *str_b = (GODataVectorStr const *)b;
return str_a->str == str_b->str && str_a->n == str_b->n;
}
static void
go_data_vector_str_load_len (GODataVector *vec)
{
vec->base.flags |= GO_DATA_VECTOR_LEN_CACHED;
vec->len = ((GODataVectorStr *)vec)->n;
}
static void
go_data_vector_str_load_values (GODataVector *vec)
{
}
static double
go_data_vector_str_get_value (GODataVector *vec, unsigned i)
{
return go_nan;
}
static char *
go_data_vector_str_get_str (GODataVector *vec, unsigned i)
{
GODataVectorStr *strs = (GODataVectorStr *)vec;
if (strs->translate_func == NULL)
return g_strdup (strs->str[i]);
return g_strdup ((strs->translate_func) (strs->str[i],
strs->translate_data));
}
static void
go_data_vector_str_class_init (GObjectClass *gobject_klass)
{
GODataClass *godata_klass = (GODataClass *) gobject_klass;
GODataVectorClass *vector_klass = (GODataVectorClass *) gobject_klass;
vector_str_parent_klass = g_type_class_peek_parent (gobject_klass);
gobject_klass->finalize = go_data_vector_str_finalize;
godata_klass->dup = go_data_vector_str_dup;
godata_klass->eq = go_data_vector_str_eq;
godata_klass->as_str = NULL;
godata_klass->from_str = NULL;
vector_klass->load_len = go_data_vector_str_load_len;
vector_klass->load_values = go_data_vector_str_load_values;
vector_klass->get_value = go_data_vector_str_get_value;
vector_klass->get_str = go_data_vector_str_get_str;
}
static void
go_data_vector_str_init (GObject *obj)
{
GODataVectorStr *str = (GODataVectorStr *)obj;
str->str = NULL;
str->n = 0;
str->translate_func = NULL;
str->translate_data = NULL;
str->translate_notify = NULL;
}
GSF_CLASS (GODataVectorStr, go_data_vector_str,
go_data_vector_str_class_init, go_data_vector_str_init,
GO_DATA_VECTOR_TYPE)
GOData *
go_data_vector_str_new (char const * const *str, unsigned n)
{
GODataVectorStr *res = g_object_new (GO_DATA_VECTOR_STR_TYPE, NULL);
res->str = str;
res->n = n;
return GO_DATA (res);
}
/**
* go_data_vector_str_set_translate_func:
* @vec: a #GODataVectorStr
* @func: a #GOTranslateFunc
* @data: data to be passed to @func and @notify
* @notify: a #GODestroyNotify function to be called when @vec is
* destroyed or when the translation function is changed
*
* Sets a function to be used for translating elements of @vec
**/
void
go_data_vector_str_set_translate_func (GODataVectorStr *vec,
GOTranslateFunc func,
gpointer data,
GDestroyNotify notify)
{
g_return_if_fail (GO_DATA_VECTOR_STR (vec) != NULL);
if (vec->translate_notify != NULL)
(*vec->translate_notify) (vec->translate_data);
vec->translate_func = func;
vec->translate_data = data;
vec->translate_notify = notify;
}
static char const *
dgettext_swapped (char const *msgid,
char const *domainname)
{
return dgettext (domainname, msgid);
}
/**
* go_data_vector_str_set_translation_domain:
* @action_group: a #GtkActionGroup
* @domain: the translation domain to use for dgettext() calls
*
* Sets the translation domain and uses dgettext() for translating the
* elements of @vec.
* Note that libgoffice expects all strings to be encoded in UTF-8, therefore
* the translation domain must have its codeset set to UTF-8, see
* bind_textdomain_codeset() in the gettext() documentation.
*
* If you're not using gettext() for localization, see
* go_data_vector_str_set_translate_func().
**/
void
go_data_vector_str_set_translation_domain (GODataVectorStr *vec,
char const *domain)
{
g_return_if_fail (GO_DATA_VECTOR_STR (vec) != NULL);
go_data_vector_str_set_translate_func (vec,
(GOTranslateFunc)dgettext_swapped, g_strdup (domain), g_free);
}