/* * Copyright (C) 2012 - Juan Ferrer Toribio * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * 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, see . */ #include "db-calc.h" #include "db-row.h" static void db_calc_param_init (GvnParamInterface * iface); /** * SECTION: db-calc * @short_description: the result of an operation made over a column or between * columns * @title: DbCalc * * A #GvnParam that holds the result of the user-given operation applied over * a given column of a #DbModel. When any of these values is modified, #DbCalc * updates its value according to the changes. * * #DbCalc has as built-in operations the sumatory and average that can be * selected passing #DB_CALC_TYPE_SUM OR #DB_CALC_TYPE_AVG, respectively. * * While creating a #DbCalc it's possible to specify either a single column * number, to calculate using only this column and no additional operations, or * a #DbCalcFunc function, to use more than one column or do any operation * needed with the #DbModel fields, in the latter case, custom data can also be * passed. **/ G_DEFINE_TYPE_WITH_CODE (DbCalc, db_calc, G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE (GVN_TYPE_PARAM, db_calc_param_init) ); /** * db_calc_new: * * Creates a new #DbCalc. * * Return value: a new #DbCalc **/ DbCalc * db_calc_new () { return g_object_new (DB_TYPE_CALC, NULL); } /** * db_calc_new_with_func: * @model: the #DbModel in which the operations will apply * @type: the #DbCalcOperationType * @func: (scope call): a #DbCalcFunc * @data: additional data that will be passed to @func * * Creates a new #DbCalc using @func as the operation that will apply to each * row of @model. * * Return value: a new #DbCalc **/ DbCalc * db_calc_new_with_func (DbModel * model, DbCalcType type, DbCalcFunc func, gpointer data) { g_return_val_if_fail (DB_IS_MODEL (model), NULL); g_return_val_if_fail (func, NULL); return g_object_new (DB_TYPE_CALC ,"model", model ,"type", type ,"function", func ,"data", data ,NULL ); } //+++++++++++++++++++++++++++++++++++++++++++++++++++ Private static void db_calc_put_value (DbCalc * self, const GValue * value) { gvn_value_copy (value, self->value); g_signal_emit_by_name (self, "value-changed", value); if (self->master) gvn_param_request_value (self->master, value, NULL); } static const GValue * db_calc_get_value (DbCalc * self) { return self->value; } static gboolean db_calc_request_value (DbCalc * self, const GValue * value, GError ** err) { return gvn_param_spec_validate (self->spec, value, err); } static GvnParam * db_calc_get_master (DbCalc * self) { return self->master; } static void db_calc_set_master (DbCalc * self, GvnParam * master) { if (self->master) g_clear_object (&self->master); if (master) self->master = g_object_ref_sink (master); } static const GvnParamSpec * db_calc_get_spec (DbCalc * self) { return self->spec; } static GvnParamStatus db_calc_get_status (DbCalc * self) { if (self->model && db_model_is_ready (self->model)) return GVN_PARAM_STATUS_OK; return GVN_PARAM_STATUS_BUSY; } static gdouble db_calc_value_to_double (const GValue * v) { switch (G_VALUE_TYPE (v)) { case G_TYPE_UINT: return (gdouble) g_value_get_uint (v); case G_TYPE_UINT64: return (gdouble) g_value_get_uint64 (v); case G_TYPE_ULONG: return (gdouble) g_value_get_ulong (v); case G_TYPE_INT: return (gdouble) g_value_get_int (v); case G_TYPE_INT64: return (gdouble) g_value_get_int64 (v); case G_TYPE_LONG: return (gdouble) g_value_get_long (v); case G_TYPE_FLOAT: return (gdouble) g_value_get_float (v); case G_TYPE_DOUBLE: return g_value_get_double (v); case G_TYPE_STRING: return g_strtod (g_value_get_string (v), NULL); default: return 0; } } static void db_calc_internal_func (DbCalc * self, DbIter * iter, GValue * out, gpointer data) { const GValue * value; if (self->column_index != -1) { value = db_model_get_value (self->model, iter, self->column_index, NULL); g_value_copy (value, g_value_init (out, G_VALUE_TYPE (value))); } else g_value_init (out, GVN_TYPE_NULL); } static void db_calc_reset (DbCalc * self) { g_free (self->column_name); self->column_name = NULL; self->column_index = -1; self->func = db_calc_internal_func; } static void db_calc_refresh (DbCalc * self) { DbIter iter; gdouble current = 0; GValue val = G_VALUE_INIT; DbModel * model = self->model; self->count = 0; if (model) { DbModelStatus status = db_model_get_status (model); if (status == DB_MODEL_STATUS_READY && self->column_name) self->column_index = db_model_get_column_index (model, self->column_name); if (status == DB_MODEL_STATUS_READY && self->column_index != -1 && db_model_get_iter_first (model, &iter)) { do { self->func (self, &iter, &val, self->data); if (!gvn_value_is_null (&val)) { current += db_calc_value_to_double (&val); self->count++; } g_value_unset (&val); } while (db_model_iter_next (model, &iter)); if (self->type == DB_CALC_TYPE_AVG) current = (self->count > 0) ? current / (gdouble) self->count : 0; } } if (self->count > 0) g_value_set_double (g_value_init (&val, G_TYPE_DOUBLE), current); else g_value_init (&val, GVN_TYPE_NULL); db_calc_put_value (self, &val); g_value_unset (&val); } static void db_calc_on_model_refresh (DbModel * model, DbModelStatus status, DbCalc * self) { db_calc_refresh (self); } static void db_calc_on_update (DbModel * model, DbIter * iter, DbCalc * self) { GValue value = G_VALUE_INIT; self->func (self, iter, &value, self->data); if (!gvn_value_is_null (&value)) { gdouble current, old; current = db_calc_value_to_double (gvn_param_get_value (GVN_PARAM (self))); old = db_calc_value_to_double (&value); if (self->type == DB_CALC_TYPE_AVG) { if (self->count != 1) current = (current * (gdouble) self->count - old) / (gdouble) (self->count - 1); else current = 0; } else if (self->type == DB_CALC_TYPE_SUM) current -= old; self->count--; g_value_unset (&value); g_value_set_double (g_value_init (&value, G_TYPE_DOUBLE), current); db_calc_put_value (self, &value); } g_value_unset (&value); } static void db_calc_on_update_after (DbModel * model, DbIter * iter, DbCalc * self) { GValue value = G_VALUE_INIT; self->func (self, iter, &value, self->data); if (!gvn_value_is_null (&value)) { gdouble current, new; current = db_calc_value_to_double (gvn_param_get_value (GVN_PARAM (self))); new = db_calc_value_to_double (&value); if (self->type == DB_CALC_TYPE_AVG) current = (current * (gdouble) self->count + new) / (gdouble) (self->count + 1); else if (self->type == DB_CALC_TYPE_SUM) current += new; self->count++; g_value_unset (&value); g_value_set_double (g_value_init (&value, G_TYPE_DOUBLE), current); db_calc_put_value (self, &value); } g_value_unset (&value); } static void db_calc_on_delete (DbModel * model, gint path, DbCalc * self) { DbIter iter; db_model_get_iter (model, &iter, path); db_calc_on_update (model, &iter, self); } //+++++++++++++++++++++++++++++++++++++++++++++++++++ Public void db_calc_set_column_name (DbCalc * self, const gchar * name) { g_return_if_fail (DB_IS_CALC (self)); db_calc_reset (self); self->column_name = g_strdup (name); db_calc_refresh (self); } void db_calc_set_column_index (DbCalc * self, gint index) { g_return_if_fail (DB_IS_CALC (self)); db_calc_reset (self); self->column_index = index; db_calc_refresh (self); } void db_calc_set_model (DbCalc * self, DbModel * model) { g_return_if_fail (DB_IS_CALC (self)); g_return_if_fail (DB_IS_MODEL (model) || !model); if (self->model) { g_object_disconnect (self->model ,"any_signal::line-updated", db_calc_on_update, self ,"any_signal::line-updated", db_calc_on_update_after, self ,"any_signal::line-deleted", db_calc_on_delete, self ,"any_signal::status-changed", db_calc_on_model_refresh, self ,NULL ); g_clear_object (&self->model); } if (model) { self->model = g_object_ref (model); g_object_connect (self->model ,"signal::line-updated", db_calc_on_update, self ,"signal-after::line-updated", db_calc_on_update_after, self ,"signal::line-deleted", db_calc_on_delete, self ,"signal::status-changed", db_calc_on_model_refresh, self ,NULL ); } db_calc_refresh (self); } //+++++++++++++++++++++++++++++++++++++++++++++++++++ Class enum { PROP_VALUE = 1 ,PROP_MASTER ,PROP_MODEL ,PROP_TYPE ,PROP_FUNC ,PROP_DATA ,PROP_COLUMN_INDEX ,PROP_COLUMN_NAME }; static void db_calc_set_property (DbCalc * self, guint id, const GValue * value, GParamSpec * pspec) { switch (id) { case PROP_VALUE: db_calc_request_value (self, g_value_get_boxed (value), NULL); break; case PROP_MASTER: db_calc_set_master (self, g_value_get_object (value)); break; case PROP_MODEL: db_calc_set_model (self, g_value_get_object (value)); break; case PROP_TYPE: self->type = g_value_get_enum (value); db_calc_refresh (self); break; case PROP_FUNC: db_calc_reset (self); if (g_value_get_pointer (value)) self->func = g_value_get_pointer (value); db_calc_refresh (self); break; case PROP_DATA: self->data = g_value_get_pointer (value); break; case PROP_COLUMN_INDEX: db_calc_set_column_index (self, g_value_get_int (value)); break; case PROP_COLUMN_NAME: db_calc_set_column_name (self, g_value_get_string (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (self, id, pspec); } } static void db_calc_get_property (DbCalc * self, guint id, GValue * value, GParamSpec * pspec) { switch (id) { case PROP_VALUE: g_value_set_boxed (value, self->value); break; case PROP_MASTER: g_value_set_object (value, self->master); break; case PROP_MODEL: g_value_set_object (value, self->model); break; case PROP_TYPE: g_value_set_enum (value, self->type); break; case PROP_FUNC: g_value_set_pointer (value, self->func); break; case PROP_DATA: g_value_set_pointer (value, self->data); break; case PROP_COLUMN_INDEX: g_value_set_int (value, self->column_index); break; case PROP_COLUMN_NAME: g_value_set_string (value, self->column_name); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (self, id, pspec); } } static void db_calc_init (DbCalc * self) { self->master = NULL; self->model = NULL; self->type = 0; self->func = NULL; self->data = NULL; self->column_index = -1; self->column_name = NULL; self->count = 0; self->spec = gvn_param_spec_new_with_attrs (G_TYPE_DOUBLE, FALSE, TRUE, NULL); self->value = g_new0 (GValue, 1); g_value_init (self->value, GVN_TYPE_NULL); } static void db_calc_finalize (DbCalc * self) { db_calc_set_model (self, NULL); db_calc_set_master (self, NULL); g_value_unset (self->value); gvn_param_spec_free (self->spec); g_free (self->value); g_free (self->column_name); G_OBJECT_CLASS (db_calc_parent_class)->finalize (G_OBJECT (self)); } static void db_calc_class_init (DbCalcClass * klass) { GObjectClass * k = G_OBJECT_CLASS (klass); k->finalize = (GObjectFinalizeFunc) db_calc_finalize; k->set_property = (GObjectSetPropertyFunc) db_calc_set_property; k->get_property = (GObjectGetPropertyFunc) db_calc_get_property; g_object_class_override_property (k, PROP_VALUE, "value"); g_object_class_override_property (k, PROP_MASTER, "master"); g_object_class_install_property (k, PROP_MODEL, g_param_spec_object ("model" ,_("Model") ,_("The model where the operations will be applied") ,DB_TYPE_MODEL ,G_PARAM_READWRITE) ); g_object_class_install_property (k, PROP_TYPE, g_param_spec_enum ("type" ,_("Operation type") ,_("The type of the operation applied over the function") ,DB_TYPE_CALC_TYPE ,DB_CALC_TYPE_SUM ,G_PARAM_READWRITE) ); g_object_class_install_property (k, PROP_FUNC, g_param_spec_pointer ("function" ,_("Function") ,_("The function to execute") ,G_PARAM_READWRITE) ); g_object_class_install_property (k, PROP_DATA, g_param_spec_pointer ("data" ,_("Data") ,_("The user provided data for the function") ,G_PARAM_READWRITE) ); g_object_class_install_property (k, PROP_COLUMN_INDEX, g_param_spec_int ("column-index" ,_("Column index") ,_("A column to apply the operations over it") ,-1 ,G_MAXINT ,-1 ,G_PARAM_READWRITE) ); g_object_class_install_property (k, PROP_COLUMN_NAME, g_param_spec_string ("column-name" ,_("Column name") ,_("A column name to apply the operations over it") ,NULL ,G_PARAM_READWRITE) ); } static void db_calc_param_init (GvnParamInterface * iface) { iface->get_value = (GvnParamGetValueFunc) db_calc_get_value; iface->request_value = (GvnParamRequestValueFunc) db_calc_request_value; iface->get_master = (GvnParamGetMasterFunc) db_calc_get_master; iface->set_master = (GvnParamSetMasterFunc) db_calc_set_master; iface->get_spec = (GvnParamGetSpecFunc) db_calc_get_spec; iface->get_status = (GvnParamGetStatusFunc) db_calc_get_status; } GType db_calc_type_get_type () { static GType type = 0; if (type == 0) { static const GEnumValue values[] = { {DB_CALC_TYPE_SUM, "DB_CALC_TYPE_SUM", "sum"} ,{DB_CALC_TYPE_AVG, "DB_CALC_TYPE_AVG", "avg"} ,{0, NULL, NULL} }; type = g_enum_register_static (g_intern_static_string ("DbCalcType"), values); } return type; }