mirror of
https://github.com/Gnucash/gnucash.git
synced 2025-02-25 18:55:30 -06:00
with the function/declaration that they substituted. Note that this doesn't use the recommended new GObject creation macros because the class names in libgnucash/engine don't follow the gnome naming convention.
1352 lines
44 KiB
C
1352 lines
44 KiB
C
/*
|
|
* gnc-tree-model-account.c -- GtkTreeModel implementation to
|
|
* display accounts in a GtkTreeView.
|
|
*
|
|
* Copyright (C) 2003 Jan Arne Petersen <jpetersen@uni-bonn.de>
|
|
* Copyright (C) 2003 David Hampton <hampton@employees.org>
|
|
*
|
|
* 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 2 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, contact:
|
|
*
|
|
* Free Software Foundation Voice: +1-617-542-5942
|
|
* 51 Franklin Street, Fifth Floor Fax: +1-617-542-2652
|
|
* Boston, MA 02110-1301, USA gnu@gnu.org
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
#include <glib/gi18n.h>
|
|
#include <string.h>
|
|
|
|
#include "gnc-tree-model-account.h"
|
|
#include "gnc-component-manager.h"
|
|
#include "Account.h"
|
|
#include "gnc-accounting-period.h"
|
|
#include "gnc-commodity.h"
|
|
#include "gnc-prefs.h"
|
|
#include "gnc-engine.h"
|
|
#include "gnc-event.h"
|
|
#include "gnc-gobject-utils.h"
|
|
#include "gnc-ui-balances.h"
|
|
#include "gnc-ui-util.h"
|
|
|
|
#define TREE_MODEL_ACCOUNT_CM_CLASS "tree-model-account"
|
|
|
|
/** Static Globals *******************************************************/
|
|
static QofLogModule log_module = GNC_MOD_GUI;
|
|
|
|
/** Declarations *********************************************************/
|
|
static void gnc_tree_model_account_class_init (GncTreeModelAccountClass *klass);
|
|
static void gnc_tree_model_account_init (GncTreeModelAccount *model);
|
|
static void gnc_tree_model_account_finalize (GObject *object);
|
|
static void gnc_tree_model_account_dispose (GObject *object);
|
|
|
|
/** Implementation of GtkTreeModel **************************************/
|
|
static void gnc_tree_model_account_tree_model_init (GtkTreeModelIface *iface);
|
|
static GtkTreeModelFlags gnc_tree_model_account_get_flags (GtkTreeModel *tree_model);
|
|
static int gnc_tree_model_account_get_n_columns (GtkTreeModel *tree_model);
|
|
static GType gnc_tree_model_account_get_column_type (GtkTreeModel *tree_model,
|
|
int index);
|
|
static gboolean gnc_tree_model_account_get_iter (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreePath *path);
|
|
static GtkTreePath *gnc_tree_model_account_get_path (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter);
|
|
static void gnc_tree_model_account_get_value (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
int column,
|
|
GValue *value);
|
|
static gboolean gnc_tree_model_account_iter_next (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter);
|
|
static gboolean gnc_tree_model_account_iter_children (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *parent);
|
|
static gboolean gnc_tree_model_account_iter_has_child (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter);
|
|
static int gnc_tree_model_account_iter_n_children (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter);
|
|
static gboolean gnc_tree_model_account_iter_nth_child (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *parent,
|
|
int n);
|
|
static gboolean gnc_tree_model_account_iter_parent (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *child);
|
|
|
|
/** Component Manager Callback ******************************************/
|
|
static void gnc_tree_model_account_event_handler (QofInstance *entity,
|
|
QofEventId event_type,
|
|
GncTreeModelAccount *model,
|
|
GncEventData *ed);
|
|
|
|
/** The instance private data for an account tree model. */
|
|
typedef struct GncTreeModelAccountPrivate
|
|
{
|
|
QofBook *book;
|
|
Account *root;
|
|
gint event_handler_id;
|
|
const gchar *negative_color;
|
|
} GncTreeModelAccountPrivate;
|
|
|
|
#define GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(o) \
|
|
((GncTreeModelAccountPrivate*)g_type_instance_get_private((GTypeInstance*)o, GNC_TYPE_TREE_MODEL_ACCOUNT))
|
|
|
|
|
|
/************************************************************/
|
|
/* Account Tree Model - Misc Functions */
|
|
/************************************************************/
|
|
static gchar*
|
|
get_negative_color (void)
|
|
{
|
|
GdkRGBA color;
|
|
GtkWidget *label = gtk_label_new ("Color");
|
|
GtkStyleContext *context = gtk_widget_get_style_context (GTK_WIDGET(label));
|
|
gtk_style_context_add_class (context, "negative-numbers");
|
|
gtk_style_context_get_color (context, GTK_STATE_FLAG_NORMAL, &color);
|
|
|
|
return gdk_rgba_to_string (&color);
|
|
}
|
|
|
|
/** Tell the GncTreeModelAccount code to update the color that it will
|
|
* use for negative numbers. This function will iterate over all
|
|
* existing models and update their setting.
|
|
*
|
|
* @internal
|
|
*/
|
|
static void
|
|
gnc_tree_model_account_update_color (gpointer gsettings, gchar *key, gpointer user_data)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
GncTreeModelAccount *model;
|
|
gboolean use_red;
|
|
|
|
g_return_if_fail(GNC_IS_TREE_MODEL_ACCOUNT(user_data));
|
|
model = user_data;
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
use_red = gnc_prefs_get_bool (GNC_PREFS_GROUP_GENERAL, GNC_PREF_NEGATIVE_IN_RED);
|
|
priv->negative_color = use_red ? get_negative_color () : NULL;
|
|
}
|
|
/************************************************************/
|
|
/* g_object required functions */
|
|
/************************************************************/
|
|
|
|
/** A pointer to the parent class of an account tree model. */
|
|
static GObjectClass *parent_class = NULL;
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GncTreeModelAccount, gnc_tree_model_account, GNC_TYPE_TREE_MODEL,
|
|
G_ADD_PRIVATE (GncTreeModelAccount)
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_TREE_MODEL,
|
|
gnc_tree_model_account_tree_model_init))
|
|
|
|
static void
|
|
gnc_tree_model_account_class_init (GncTreeModelAccountClass *klass)
|
|
{
|
|
GObjectClass *o_class;
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
o_class = G_OBJECT_CLASS (klass);
|
|
|
|
/* GObject signals */
|
|
o_class->finalize = gnc_tree_model_account_finalize;
|
|
o_class->dispose = gnc_tree_model_account_dispose;
|
|
}
|
|
|
|
static void
|
|
gnc_tree_model_account_init (GncTreeModelAccount *model)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
gboolean red;
|
|
|
|
ENTER("model %p", model);
|
|
while (model->stamp == 0)
|
|
{
|
|
model->stamp = g_random_int ();
|
|
}
|
|
|
|
red = gnc_prefs_get_bool(GNC_PREFS_GROUP_GENERAL, GNC_PREF_NEGATIVE_IN_RED);
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
priv->book = NULL;
|
|
priv->root = NULL;
|
|
priv->negative_color = red ? get_negative_color () : NULL;
|
|
|
|
gnc_prefs_register_cb(GNC_PREFS_GROUP_GENERAL, GNC_PREF_NEGATIVE_IN_RED,
|
|
gnc_tree_model_account_update_color,
|
|
model);
|
|
|
|
LEAVE(" ");
|
|
}
|
|
|
|
static void
|
|
gnc_tree_model_account_finalize (GObject *object)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
GncTreeModelAccount *model;
|
|
|
|
g_return_if_fail (object != NULL);
|
|
g_return_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (object));
|
|
|
|
ENTER("model %p", object);
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (object);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
priv->book = NULL;
|
|
|
|
if (G_OBJECT_CLASS (parent_class)->finalize)
|
|
G_OBJECT_CLASS(parent_class)->finalize (object);
|
|
LEAVE(" ");
|
|
}
|
|
|
|
static void
|
|
gnc_tree_model_account_dispose (GObject *object)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
GncTreeModelAccount *model;
|
|
|
|
g_return_if_fail (object != NULL);
|
|
g_return_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (object));
|
|
|
|
ENTER("model %p", object);
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (object);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
if (priv->event_handler_id)
|
|
{
|
|
qof_event_unregister_handler (priv->event_handler_id);
|
|
priv->event_handler_id = 0;
|
|
}
|
|
|
|
gnc_prefs_remove_cb_by_func(GNC_PREFS_GROUP_GENERAL, GNC_PREF_NEGATIVE_IN_RED,
|
|
gnc_tree_model_account_update_color,
|
|
model);
|
|
|
|
if (G_OBJECT_CLASS (parent_class)->dispose)
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
LEAVE(" ");
|
|
}
|
|
|
|
|
|
/************************************************************/
|
|
/* New Model Creation */
|
|
/************************************************************/
|
|
|
|
GtkTreeModel *
|
|
gnc_tree_model_account_new (Account *root)
|
|
{
|
|
GncTreeModelAccount *model;
|
|
GncTreeModelAccountPrivate *priv;
|
|
const GList *item;
|
|
|
|
ENTER("root %p", root);
|
|
item = gnc_gobject_tracking_get_list(GNC_TREE_MODEL_ACCOUNT_NAME);
|
|
for ( ; item; item = g_list_next(item))
|
|
{
|
|
model = (GncTreeModelAccount *)item->data;
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
if (priv->root == root)
|
|
{
|
|
g_object_ref(G_OBJECT(model));
|
|
LEAVE("returning existing model %p", model);
|
|
return GTK_TREE_MODEL(model);
|
|
}
|
|
}
|
|
|
|
model = g_object_new (GNC_TYPE_TREE_MODEL_ACCOUNT,
|
|
NULL);
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
priv->book = gnc_get_current_book();
|
|
priv->root = root;
|
|
|
|
priv->event_handler_id = qof_event_register_handler
|
|
((QofEventHandler)gnc_tree_model_account_event_handler, model);
|
|
|
|
LEAVE("model %p", model);
|
|
return GTK_TREE_MODEL (model);
|
|
}
|
|
|
|
|
|
/************************************************************/
|
|
/* Gnc Tree Model Debugging Utility Function */
|
|
/************************************************************/
|
|
|
|
#define ITER_STRING_LEN 128
|
|
|
|
static const gchar *
|
|
iter_to_string (GtkTreeIter *iter)
|
|
{
|
|
#ifdef G_THREADS_ENABLED
|
|
static GPrivate gtmits_buffer_key = G_PRIVATE_INIT(g_free);
|
|
gchar *string;
|
|
|
|
string = g_private_get (>mits_buffer_key);
|
|
if (string == NULL)
|
|
{
|
|
string = g_malloc(ITER_STRING_LEN + 1);
|
|
g_private_set (>mits_buffer_key, string);
|
|
}
|
|
#else
|
|
static char string[ITER_STRING_LEN + 1];
|
|
#endif
|
|
|
|
if (iter)
|
|
snprintf(string, ITER_STRING_LEN,
|
|
"[stamp:%x data:%p (%s), %p, %d]",
|
|
iter->stamp, iter->user_data,
|
|
xaccAccountGetName ((Account *) iter->user_data),
|
|
iter->user_data2, GPOINTER_TO_INT(iter->user_data3));
|
|
else
|
|
strcpy(string, "(null)");
|
|
return string;
|
|
}
|
|
|
|
|
|
/************************************************************/
|
|
/* Gtk Tree Model Required Interface Functions */
|
|
/************************************************************/
|
|
|
|
static void
|
|
gnc_tree_model_account_tree_model_init (GtkTreeModelIface *iface)
|
|
{
|
|
iface->get_flags = gnc_tree_model_account_get_flags;
|
|
iface->get_n_columns = gnc_tree_model_account_get_n_columns;
|
|
iface->get_column_type = gnc_tree_model_account_get_column_type;
|
|
iface->get_iter = gnc_tree_model_account_get_iter;
|
|
iface->get_path = gnc_tree_model_account_get_path;
|
|
iface->get_value = gnc_tree_model_account_get_value;
|
|
iface->iter_next = gnc_tree_model_account_iter_next;
|
|
iface->iter_children = gnc_tree_model_account_iter_children;
|
|
iface->iter_has_child = gnc_tree_model_account_iter_has_child;
|
|
iface->iter_n_children = gnc_tree_model_account_iter_n_children;
|
|
iface->iter_nth_child = gnc_tree_model_account_iter_nth_child;
|
|
iface->iter_parent = gnc_tree_model_account_iter_parent;
|
|
}
|
|
|
|
static GtkTreeModelFlags
|
|
gnc_tree_model_account_get_flags (GtkTreeModel *tree_model)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
gnc_tree_model_account_get_n_columns (GtkTreeModel *tree_model)
|
|
{
|
|
g_return_val_if_fail(GNC_IS_TREE_MODEL_ACCOUNT(tree_model), -1);
|
|
|
|
return GNC_TREE_MODEL_ACCOUNT_NUM_COLUMNS;
|
|
}
|
|
|
|
static GType
|
|
gnc_tree_model_account_get_column_type (GtkTreeModel *tree_model,
|
|
int index)
|
|
{
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), G_TYPE_INVALID);
|
|
g_return_val_if_fail ((index < GNC_TREE_MODEL_ACCOUNT_NUM_COLUMNS) && (index >= 0), G_TYPE_INVALID);
|
|
|
|
switch (index)
|
|
{
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_NAME:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TYPE:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COMMODITY:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CODE:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_DESCRIPTION:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PRESENT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PRESENT_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE_PERIOD:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CLEARED:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CLEARED_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED_DATE:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_FUTURE_MIN:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_FUTURE_MIN_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL_REPORT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL_PERIOD:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_NOTES:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TAX_INFO:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TAX_INFO_SUB_ACCT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_LASTNUM:
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_PRESENT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_ACCOUNT:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_BALANCE:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_BALANCE_PERIOD:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_CLEARED:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_RECONCILED:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_FUTURE_MIN:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_TOTAL:
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_TOTAL_PERIOD:
|
|
return G_TYPE_STRING;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PLACEHOLDER:
|
|
return G_TYPE_BOOLEAN;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
return G_TYPE_INVALID;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_get_iter (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreePath *path)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
GncTreeModelAccount *model;
|
|
Account *account, *parent;
|
|
gint i, *indices;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
|
|
{
|
|
gchar *path_string = gtk_tree_path_to_string(path);
|
|
ENTER("model %p, iter %p, path %s", tree_model, iter, path_string);
|
|
g_free(path_string);
|
|
}
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
if (gtk_tree_path_get_depth (path) <= 0)
|
|
{
|
|
LEAVE("bad depth");
|
|
return FALSE;
|
|
}
|
|
|
|
indices = gtk_tree_path_get_indices (path);
|
|
if (indices[0] != 0)
|
|
{
|
|
LEAVE("bad root index");
|
|
return FALSE;
|
|
}
|
|
|
|
parent = NULL;
|
|
account = priv->root;
|
|
for (i = 1; i < gtk_tree_path_get_depth (path); i++)
|
|
{
|
|
parent = account;
|
|
account = gnc_account_nth_child(parent, indices[i]);
|
|
if (account == NULL)
|
|
{
|
|
iter->stamp = 0;
|
|
LEAVE("bad index");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
iter->stamp = model->stamp;
|
|
iter->user_data = account;
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (indices[i - 1]);
|
|
|
|
LEAVE("iter %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
static GtkTreePath *
|
|
gnc_tree_model_account_get_path (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GncTreeModelAccount *model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
GncTreeModelAccountPrivate *priv;
|
|
Account *account, *parent;
|
|
GtkTreePath *path;
|
|
gint i;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model), NULL);
|
|
g_return_val_if_fail (iter != NULL, NULL);
|
|
g_return_val_if_fail (iter->user_data != NULL, NULL);
|
|
g_return_val_if_fail (iter->stamp == model->stamp, NULL);
|
|
|
|
ENTER("model %p, iter %s", model, iter_to_string(iter));
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
if (priv->root == NULL)
|
|
{
|
|
LEAVE("failed (1)");
|
|
return NULL;
|
|
}
|
|
|
|
account = (Account *) iter->user_data;
|
|
parent = (Account *) iter->user_data2;
|
|
|
|
path = gtk_tree_path_new ();
|
|
while (parent)
|
|
{
|
|
i = gnc_account_child_index(parent, account);
|
|
if (i == -1)
|
|
{
|
|
gtk_tree_path_free (path);
|
|
LEAVE("failed (3)");
|
|
return NULL;
|
|
}
|
|
gtk_tree_path_prepend_index (path, i);
|
|
account = parent;
|
|
parent = gnc_account_get_parent(account);
|
|
};
|
|
|
|
/* Add the root node. */
|
|
gtk_tree_path_prepend_index (path, 0);
|
|
|
|
{
|
|
gchar *path_string = gtk_tree_path_to_string(path);
|
|
LEAVE("path (4) %s", path_string);
|
|
g_free(path_string);
|
|
}
|
|
return path;
|
|
}
|
|
|
|
static void
|
|
gnc_tree_model_account_set_color(GncTreeModelAccount *model,
|
|
gboolean negative,
|
|
GValue *value)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
if (negative)
|
|
g_value_set_static_string (value, priv->negative_color);
|
|
else
|
|
g_value_set_static_string (value, NULL);
|
|
}
|
|
|
|
static gchar *
|
|
gnc_tree_model_account_compute_period_balance(GncTreeModelAccount *model,
|
|
Account *acct,
|
|
gboolean recurse,
|
|
gboolean *negative)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
time64 t1, t2;
|
|
gnc_numeric b3;
|
|
|
|
if ( negative )
|
|
*negative = FALSE;
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
if (acct == priv->root)
|
|
return g_strdup("");
|
|
|
|
t1 = gnc_accounting_period_fiscal_start();
|
|
t2 = gnc_accounting_period_fiscal_end();
|
|
|
|
if (t1 > t2)
|
|
return g_strdup("");
|
|
|
|
b3 = xaccAccountGetBalanceChangeForPeriod(acct, t1, t2, recurse);
|
|
if (gnc_reverse_balance (acct))
|
|
b3 = gnc_numeric_neg (b3);
|
|
|
|
if (negative)
|
|
*negative = gnc_numeric_negative_p(b3);
|
|
|
|
return g_strdup(xaccPrintAmount(b3, gnc_account_print_info(acct, TRUE)));
|
|
}
|
|
|
|
static void
|
|
gnc_tree_model_account_get_value (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
int column,
|
|
GValue *value)
|
|
{
|
|
GncTreeModelAccount *model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
GncTreeModelAccountPrivate *priv;
|
|
Account *account;
|
|
gboolean negative; /* used to set "deficit style" also known as red numbers */
|
|
gchar *string;
|
|
time64 last_date;
|
|
|
|
g_return_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model));
|
|
g_return_if_fail (iter != NULL);
|
|
g_return_if_fail (iter->user_data != NULL);
|
|
g_return_if_fail (iter->stamp == model->stamp);
|
|
|
|
ENTER("model %p, iter %s, col %d", tree_model,
|
|
iter_to_string(iter), column);
|
|
|
|
account = (Account *) iter->user_data;
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
switch (column)
|
|
{
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_NAME:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
if (account == priv->root)
|
|
g_value_set_string (value, _("New top level account"));
|
|
else
|
|
g_value_set_string (value, xaccAccountGetName (account));
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TYPE:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value,
|
|
xaccAccountGetTypeStr (xaccAccountGetType (account)));
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CODE:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value, xaccAccountGetCode (account));
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COMMODITY:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value,
|
|
gnc_commodity_get_fullname(xaccAccountGetCommodity (account)));
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_DESCRIPTION:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value, xaccAccountGetDescription (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PRESENT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetPresentBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PRESENT_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetPresentBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_PRESENT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetPresentBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free(string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetBalanceInCurrency,
|
|
account, FALSE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetBalanceInCurrency,
|
|
account, FALSE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_BALANCE:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetBalanceInCurrency,
|
|
account, FALSE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free(string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_BALANCE_PERIOD:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_tree_model_account_compute_period_balance(model, account, FALSE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_BALANCE_PERIOD:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_tree_model_account_compute_period_balance(model, account, FALSE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free (string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CLEARED:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetClearedBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_CLEARED_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetClearedBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_CLEARED:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetClearedBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free(string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetReconciledBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetReconciledBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_RECONCILED_DATE:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
if (xaccAccountGetReconcileLastDate(account, &last_date))
|
|
{
|
|
g_value_take_string(value, qof_print_date(last_date));
|
|
}
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_RECONCILED:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetReconciledBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free (string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_FUTURE_MIN:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetProjectedMinimumBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_FUTURE_MIN_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetProjectedMinimumBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_FUTURE_MIN:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetProjectedMinimumBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free (string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL_REPORT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_report_balance(xaccAccountGetBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_TOTAL:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_ui_account_get_print_balance(xaccAccountGetBalanceInCurrency,
|
|
account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free (string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TOTAL_PERIOD:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_tree_model_account_compute_period_balance(model, account, TRUE, &negative);
|
|
g_value_take_string (value, string);
|
|
break;
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_TOTAL_PERIOD:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
string = gnc_tree_model_account_compute_period_balance(model, account, TRUE, &negative);
|
|
gnc_tree_model_account_set_color(model, negative, value);
|
|
g_free (string);
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_COLOR_ACCOUNT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value, xaccAccountGetColor (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_NOTES:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value, xaccAccountGetNotes (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TAX_INFO:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_take_string (value, gnc_ui_account_get_tax_info_string (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_TAX_INFO_SUB_ACCT:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_take_string (value, gnc_ui_account_get_tax_info_sub_acct_string (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_LASTNUM:
|
|
g_value_init (value, G_TYPE_STRING);
|
|
g_value_set_string (value, xaccAccountGetLastNum (account));
|
|
break;
|
|
|
|
case GNC_TREE_MODEL_ACCOUNT_COL_PLACEHOLDER:
|
|
g_value_init (value, G_TYPE_BOOLEAN);
|
|
g_value_set_boolean (value, xaccAccountGetPlaceholder (account));
|
|
break;
|
|
|
|
default:
|
|
g_assert_not_reached ();
|
|
break;
|
|
}
|
|
LEAVE(" ");
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_iter_next (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GncTreeModelAccount *model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
Account *account, *parent;
|
|
gint i;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model), FALSE);
|
|
g_return_val_if_fail (iter != NULL, FALSE);
|
|
g_return_val_if_fail (iter->user_data != NULL, FALSE);
|
|
g_return_val_if_fail (iter->stamp == model->stamp, FALSE);
|
|
|
|
ENTER("model %p, iter %s", tree_model, iter_to_string(iter));
|
|
|
|
parent = (Account *) iter->user_data2;
|
|
if (parent == NULL)
|
|
{
|
|
/* This is the root. There is no next. */
|
|
LEAVE("at root");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Get the *next* sibling account. */
|
|
i = GPOINTER_TO_INT (iter->user_data3);
|
|
account = gnc_account_nth_child (parent, i + 1);
|
|
if (account == NULL)
|
|
{
|
|
iter->stamp = 0;
|
|
LEAVE("failed (3)");
|
|
return FALSE;
|
|
}
|
|
|
|
iter->user_data = account;
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (i + 1);
|
|
|
|
LEAVE("iter %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_iter_children (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *parent_iter)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
GncTreeModelAccount *model;
|
|
Account *account, *parent;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
ENTER("model %p, iter %p (to be filed in), parent %s",
|
|
tree_model, iter, (parent_iter ? iter_to_string(parent_iter) : "(null)"));
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
if (priv->root == NULL)
|
|
{
|
|
iter->stamp = 0;
|
|
LEAVE("failed (no root)");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Special case when no parent supplied. */
|
|
if (!parent_iter)
|
|
{
|
|
iter->user_data = priv->root;
|
|
iter->user_data2 = NULL;
|
|
iter->user_data3 = GINT_TO_POINTER (0);
|
|
iter->stamp = model->stamp;
|
|
LEAVE("iter (2) %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
gnc_leave_return_val_if_fail (parent_iter != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (parent_iter->user_data != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (parent_iter->stamp == model->stamp, FALSE);
|
|
|
|
parent = (Account *)parent_iter->user_data;
|
|
account = gnc_account_nth_child (parent, 0);
|
|
|
|
if (account == NULL)
|
|
{
|
|
iter->stamp = 0;
|
|
LEAVE("failed (child account is null)");
|
|
return FALSE;
|
|
}
|
|
|
|
iter->user_data = account;
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (0);
|
|
iter->stamp = model->stamp;
|
|
LEAVE("iter (3) %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_iter_has_child (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GncTreeModelAccount *model;
|
|
Account *account;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
|
|
ENTER("model %p, iter %s", tree_model, iter_to_string(iter));
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
|
|
gnc_leave_return_val_if_fail (iter != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (iter->user_data != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (iter->stamp == model->stamp, FALSE);
|
|
|
|
account = (Account *) iter->user_data;
|
|
if (gnc_account_n_children(account) > 0)
|
|
{
|
|
LEAVE("yes");
|
|
return TRUE;
|
|
}
|
|
|
|
LEAVE("no");
|
|
return FALSE;
|
|
}
|
|
|
|
static int
|
|
gnc_tree_model_account_iter_n_children (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GncTreeModelAccount *model;
|
|
gint num;
|
|
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
ENTER("model %p, iter %s", tree_model, iter_to_string(iter));
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
|
|
if (iter == NULL)
|
|
{
|
|
/* How many children does the invisible root node
|
|
* have. One! Its the real root account node. */
|
|
LEAVE("count is 1");
|
|
return 1;
|
|
}
|
|
|
|
gnc_leave_return_val_if_fail (iter != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (iter->user_data != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (iter->stamp == model->stamp, FALSE);
|
|
|
|
num = gnc_account_n_children(iter->user_data);
|
|
LEAVE("count is %d", num);
|
|
return num;
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_iter_nth_child (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *parent_iter,
|
|
int n)
|
|
{
|
|
GncTreeModelAccount *model;
|
|
GncTreeModelAccountPrivate *priv;
|
|
Account *account, *parent;
|
|
|
|
if (parent_iter)
|
|
{
|
|
gchar *parent_string;
|
|
parent_string = g_strdup(iter_to_string(parent_iter));
|
|
ENTER("model %p, iter %s, parent_iter %s, n %d",
|
|
tree_model, iter_to_string(iter),
|
|
parent_string, n);
|
|
g_free(parent_string);
|
|
}
|
|
else
|
|
{
|
|
ENTER("model %p, iter %s, parent_iter (null), n %d",
|
|
tree_model, iter_to_string(iter), n);
|
|
}
|
|
gnc_leave_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
/* Special case when no parent supplied. */
|
|
if (!parent_iter)
|
|
{
|
|
if (n != 0)
|
|
{
|
|
LEAVE("bad root index");
|
|
return FALSE;
|
|
}
|
|
|
|
iter->user_data = priv->root;
|
|
iter->user_data2 = NULL;
|
|
iter->user_data3 = GINT_TO_POINTER (0);
|
|
iter->stamp = model->stamp;
|
|
LEAVE("root %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
gnc_leave_return_val_if_fail (parent_iter->user_data != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (parent_iter->stamp == model->stamp, FALSE);
|
|
|
|
parent = (Account *)parent_iter->user_data;
|
|
account = gnc_account_nth_child(parent, n);
|
|
if (account == NULL)
|
|
{
|
|
iter->stamp = 0;
|
|
LEAVE("failed (2)");
|
|
return FALSE;
|
|
}
|
|
|
|
iter->user_data = account;
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (n);
|
|
iter->stamp = model->stamp;
|
|
LEAVE("iter (2) %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gnc_tree_model_account_iter_parent (GtkTreeModel *tree_model,
|
|
GtkTreeIter *iter,
|
|
GtkTreeIter *child)
|
|
{
|
|
GncTreeModelAccount *model;
|
|
Account *account, *parent;
|
|
gint i;
|
|
|
|
if (child)
|
|
{
|
|
gchar *child_string;
|
|
|
|
child_string = g_strdup(iter_to_string(child));
|
|
ENTER("model %p, iter %s, child %s",
|
|
tree_model, iter_to_string(iter),
|
|
child_string);
|
|
g_free(child_string);
|
|
}
|
|
else
|
|
{
|
|
ENTER("model %p, iter %s, child (null)",
|
|
tree_model, iter_to_string(iter));
|
|
}
|
|
gnc_leave_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (tree_model), FALSE);
|
|
|
|
model = GNC_TREE_MODEL_ACCOUNT (tree_model);
|
|
|
|
gnc_leave_return_val_if_fail (child != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (child->user_data != NULL, FALSE);
|
|
gnc_leave_return_val_if_fail (child->stamp == model->stamp, FALSE);
|
|
|
|
account = (Account *) child->user_data;
|
|
account = gnc_account_get_parent(account);
|
|
if (account == NULL)
|
|
{
|
|
/* Can't go up from the root node */
|
|
iter->stamp = 0;
|
|
LEAVE("failed (1)");
|
|
return FALSE;
|
|
}
|
|
|
|
parent = gnc_account_get_parent(account);
|
|
if (parent == NULL)
|
|
{
|
|
/* Now at the root. */
|
|
i = 0;
|
|
}
|
|
else
|
|
{
|
|
i = gnc_account_child_index(parent, account);
|
|
}
|
|
iter->user_data = account;
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (i);
|
|
iter->stamp = model->stamp;
|
|
LEAVE("iter (2) %s", iter_to_string(iter));
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/************************************************************/
|
|
/* Account Tree View Filter Functions */
|
|
/************************************************************/
|
|
|
|
/*
|
|
* Convert a model/iter pair to a gnucash account. This routine should
|
|
* only be called from an account tree view filter function.
|
|
*/
|
|
Account *
|
|
gnc_tree_model_account_get_account (GncTreeModelAccount *model,
|
|
GtkTreeIter *iter)
|
|
{
|
|
g_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model), NULL);
|
|
g_return_val_if_fail (iter != NULL, NULL);
|
|
g_return_val_if_fail (iter->user_data != NULL, NULL);
|
|
g_return_val_if_fail (iter->stamp == model->stamp, NULL);
|
|
|
|
return (Account *) iter->user_data;
|
|
}
|
|
|
|
/*
|
|
* Convert a model/account pair into a gtk_tree_model_iter. This
|
|
* routine should only be called from the file
|
|
* gnc-tree-view-account.c.
|
|
*/
|
|
gboolean
|
|
gnc_tree_model_account_get_iter_from_account (GncTreeModelAccount *model,
|
|
Account *account,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
Account *parent;
|
|
gint i;
|
|
|
|
ENTER("model %p, account %p, iter %p", model, account, iter);
|
|
gnc_leave_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model), FALSE);
|
|
gnc_leave_return_val_if_fail ((account != NULL), FALSE);
|
|
gnc_leave_return_val_if_fail ((iter != NULL), FALSE);
|
|
|
|
iter->user_data = account;
|
|
iter->stamp = model->stamp;
|
|
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
if (account == priv->root)
|
|
{
|
|
iter->user_data2 = NULL;
|
|
iter->user_data3 = GINT_TO_POINTER (0);
|
|
LEAVE("Matched root");
|
|
return TRUE;
|
|
}
|
|
|
|
if (priv->root != gnc_account_get_root (account))
|
|
{
|
|
LEAVE("Root doesn't match");
|
|
return FALSE;
|
|
}
|
|
|
|
parent = gnc_account_get_parent(account);
|
|
i = gnc_account_child_index(parent, account);
|
|
iter->user_data2 = parent;
|
|
iter->user_data3 = GINT_TO_POINTER (i);
|
|
LEAVE("iter %s", iter_to_string(iter));
|
|
return (i != -1);
|
|
}
|
|
|
|
/*
|
|
* Convert a model/account pair into a gtk_tree_model_path. This
|
|
* routine should only be called from the file
|
|
* gnc-tree-view-account.c.
|
|
*/
|
|
GtkTreePath *
|
|
gnc_tree_model_account_get_path_from_account (GncTreeModelAccount *model,
|
|
Account *account)
|
|
{
|
|
GtkTreeIter tree_iter;
|
|
GtkTreePath *tree_path;
|
|
|
|
ENTER("model %p, account %p", model, account);
|
|
gnc_leave_return_val_if_fail (GNC_IS_TREE_MODEL_ACCOUNT (model), NULL);
|
|
gnc_leave_return_val_if_fail (account != NULL, NULL);
|
|
|
|
if (!gnc_tree_model_account_get_iter_from_account (model, account,
|
|
&tree_iter))
|
|
{
|
|
LEAVE("no iter");
|
|
return NULL;
|
|
}
|
|
|
|
tree_path = gtk_tree_model_get_path (GTK_TREE_MODEL(model), &tree_iter);
|
|
if (tree_path)
|
|
{
|
|
gchar *path_string = gtk_tree_path_to_string(tree_path);
|
|
LEAVE("path (2) %s", path_string);
|
|
g_free(path_string);
|
|
}
|
|
else
|
|
{
|
|
LEAVE("no path");
|
|
}
|
|
return tree_path;
|
|
}
|
|
|
|
/************************************************************/
|
|
/* Account Tree Model - Engine Event Handling Functions */
|
|
/************************************************************/
|
|
|
|
static void
|
|
increment_stamp(GncTreeModelAccount *model)
|
|
{
|
|
do model->stamp++;
|
|
while (!model->stamp);
|
|
}
|
|
|
|
static void
|
|
propagate_change(GtkTreeModel *model, GtkTreePath *path, gint toggle_if_num)
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
/* Already at the invisible root node? */
|
|
if (!gtk_tree_path_up(path))
|
|
return;
|
|
|
|
/* Did we just move up to the invisible root node? */
|
|
if (gtk_tree_path_get_depth(path) == 0)
|
|
return;
|
|
|
|
/* Handle the immediate parent */
|
|
if (gtk_tree_model_get_iter(model, &iter, path))
|
|
{
|
|
gtk_tree_model_row_changed(model, path, &iter);
|
|
if (gtk_tree_model_iter_n_children(model, &iter) == toggle_if_num)
|
|
gtk_tree_model_row_has_child_toggled(model, path, &iter);
|
|
}
|
|
|
|
/* All other ancestors */
|
|
while (gtk_tree_path_up(path) && gtk_tree_path_get_depth(path) > 0 &&
|
|
gtk_tree_model_get_iter(model, &iter, path))
|
|
{
|
|
gtk_tree_model_row_changed(model, path, &iter);
|
|
}
|
|
}
|
|
|
|
/** This function is the handler for all event messages from the
|
|
* engine. Its purpose is to update the account tree model any time
|
|
* an account is added to the engine or deleted from the engine.
|
|
* This change to the model is then propagated to any/all overlying
|
|
* filters and views. This function listens to the ADD, REMOVE, and
|
|
* DESTROY events.
|
|
*
|
|
* @internal
|
|
*
|
|
* @warning There is a "Catch 22" situation here.
|
|
* gtk_tree_model_row_deleted() can't be called until after the item
|
|
* has been deleted from the real model (which is the engine's
|
|
* account tree for us), but once the account has been deleted from
|
|
* the engine we have no way to determine the path to pass to
|
|
* row_deleted(). This is a PITA, but the only other choice is to
|
|
* have this model mirror the engine's accounts instead of
|
|
* referencing them directly.
|
|
*
|
|
* @param entity The guid of the affected item.
|
|
*
|
|
* @param type The type of the affected item. This function only
|
|
* cares about items of type "account".
|
|
*
|
|
* @param event type The type of the event. This function only cares
|
|
* about items of type ADD, REMOVE, MODIFY, and DESTROY.
|
|
*
|
|
* @param user_data A pointer to the account tree model.
|
|
*/
|
|
static void
|
|
gnc_tree_model_account_event_handler (QofInstance *entity,
|
|
QofEventId event_type,
|
|
GncTreeModelAccount *model,
|
|
GncEventData *ed)
|
|
{
|
|
GncTreeModelAccountPrivate *priv;
|
|
const gchar *parent_name;
|
|
GtkTreePath *path = NULL;
|
|
GtkTreeIter iter;
|
|
Account *account, *parent;
|
|
|
|
g_return_if_fail(model); /* Required */
|
|
if (!GNC_IS_ACCOUNT(entity))
|
|
return;
|
|
|
|
ENTER("entity %p of type %d, model %p, event_data %p",
|
|
entity, event_type, model, ed);
|
|
priv = GNC_TREE_MODEL_ACCOUNT_GET_PRIVATE(model);
|
|
|
|
account = GNC_ACCOUNT(entity);
|
|
if (gnc_account_get_book(account) != priv->book)
|
|
{
|
|
LEAVE("not in this book");
|
|
return;
|
|
}
|
|
if (gnc_account_get_root(account) != priv->root)
|
|
{
|
|
LEAVE("not in this model");
|
|
return;
|
|
}
|
|
/* What to do, that to do. */
|
|
switch (event_type)
|
|
{
|
|
case QOF_EVENT_ADD:
|
|
/* Tell the filters/views where the new account was added. */
|
|
DEBUG("add account %p (%s)", account, xaccAccountGetName(account));
|
|
path = gnc_tree_model_account_get_path_from_account(model, account);
|
|
if (!path)
|
|
{
|
|
DEBUG("can't generate path");
|
|
break;
|
|
}
|
|
increment_stamp(model);
|
|
if (!gnc_tree_model_account_get_iter(GTK_TREE_MODEL(model), &iter, path))
|
|
{
|
|
DEBUG("can't generate iter");
|
|
break;
|
|
}
|
|
gtk_tree_model_row_inserted (GTK_TREE_MODEL(model), path, &iter);
|
|
propagate_change(GTK_TREE_MODEL(model), path, 1);
|
|
break;
|
|
|
|
case QOF_EVENT_REMOVE:
|
|
if (!ed) /* Required for a remove. */
|
|
break;
|
|
parent = ed->node ? GNC_ACCOUNT(ed->node) : priv->root;
|
|
parent_name = ed->node ? xaccAccountGetName(parent) : "Root";
|
|
DEBUG("remove child %d of account %p (%s)", ed->idx, parent, parent_name);
|
|
path = gnc_tree_model_account_get_path_from_account(model, parent);
|
|
if (!path)
|
|
{
|
|
DEBUG("can't generate path");
|
|
break;
|
|
}
|
|
increment_stamp(model);
|
|
gtk_tree_path_append_index (path, ed->idx);
|
|
gtk_tree_model_row_deleted (GTK_TREE_MODEL(model), path);
|
|
propagate_change(GTK_TREE_MODEL(model), path, 0);
|
|
break;
|
|
|
|
case QOF_EVENT_MODIFY:
|
|
DEBUG("modify account %p (%s)", account, xaccAccountGetName(account));
|
|
path = gnc_tree_model_account_get_path_from_account(model, account);
|
|
if (!path)
|
|
{
|
|
DEBUG("can't generate path");
|
|
break;
|
|
}
|
|
if (!gnc_tree_model_account_get_iter(GTK_TREE_MODEL(model), &iter, path))
|
|
{
|
|
DEBUG("can't generate iter");
|
|
break;
|
|
}
|
|
gtk_tree_model_row_changed(GTK_TREE_MODEL(model), path, &iter);
|
|
propagate_change(GTK_TREE_MODEL(model), path, -1);
|
|
break;
|
|
|
|
default:
|
|
LEAVE("unknown event type");
|
|
return;
|
|
}
|
|
|
|
if (path)
|
|
gtk_tree_path_free(path);
|
|
LEAVE(" ");
|
|
return;
|
|
}
|