mirror of
https://github.com/Gnucash/gnucash.git
synced 2024-12-02 05:29:20 -06:00
839 lines
25 KiB
C
839 lines
25 KiB
C
/********************************************************************\
|
|
* gnc-budget.c -- Implementation of the top level Budgeting API. *
|
|
* Copyright (C) 04 sep 2003 Darin Willits <darin@willits.ca> *
|
|
* Copyright (C) 2005-2006 Chris Shoemaker <c.shoemaker@cox.net> *
|
|
* *
|
|
* 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 <glib.h>
|
|
#include <glib/gprintf.h>
|
|
#include <glib/gi18n.h>
|
|
#include <time.h>
|
|
#include <qof.h>
|
|
#include <qofbookslots.h>
|
|
#include <qofinstance-p.h>
|
|
|
|
#include "Account.h"
|
|
|
|
#include "gnc-budget.h"
|
|
#include "gnc-commodity.h"
|
|
|
|
static QofLogModule log_module = GNC_MOD_ENGINE;
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_NAME, /* Table */
|
|
PROP_DESCRIPTION, /* Table */
|
|
PROP_NUM_PERIODS, /* Table */
|
|
PROP_RUNTIME_0,
|
|
PROP_RECURRENCE, /* Cached pointer; Recurrence table holds budget guid */
|
|
};
|
|
|
|
struct budget_s
|
|
{
|
|
QofInstance inst;
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
QofInstanceClass parent_class;
|
|
} BudgetClass;
|
|
|
|
typedef struct GncBudgetPrivate
|
|
{
|
|
/* The name is an arbitrary string assigned by the user. */
|
|
const gchar *name;
|
|
|
|
/* The description is an arbitrary string assigned by the user. */
|
|
const gchar *description;
|
|
|
|
/* Recurrence (period info) for the budget */
|
|
Recurrence recurrence;
|
|
|
|
/* Number of periods */
|
|
guint num_periods;
|
|
} GncBudgetPrivate;
|
|
|
|
#define GET_PRIVATE(o) \
|
|
((GncBudgetPrivate*)g_type_instance_get_private((GTypeInstance*)o, GNC_TYPE_BUDGET))
|
|
|
|
struct _GncBudgetClass
|
|
{
|
|
QofInstanceClass parent_class;
|
|
};
|
|
|
|
/* GObject Initialization */
|
|
G_DEFINE_TYPE_WITH_PRIVATE(GncBudget, gnc_budget, QOF_TYPE_INSTANCE)
|
|
|
|
static void
|
|
gnc_budget_init(GncBudget* budget)
|
|
{
|
|
GncBudgetPrivate* priv;
|
|
GDate *date;
|
|
|
|
priv = GET_PRIVATE(budget);
|
|
priv->name = CACHE_INSERT(_("Unnamed Budget"));
|
|
priv->description = CACHE_INSERT("");
|
|
|
|
priv->num_periods = 12;
|
|
date = gnc_g_date_new_today ();
|
|
g_date_subtract_days(date, g_date_get_day(date) - 1);
|
|
recurrenceSet(&priv->recurrence, 1, PERIOD_MONTH, date, WEEKEND_ADJ_NONE);
|
|
g_date_free (date);
|
|
}
|
|
|
|
static void
|
|
gnc_budget_dispose (GObject *budgetp)
|
|
{
|
|
G_OBJECT_CLASS(gnc_budget_parent_class)->dispose(budgetp);
|
|
}
|
|
|
|
static void
|
|
gnc_budget_finalize(GObject* budgetp)
|
|
{
|
|
G_OBJECT_CLASS(gnc_budget_parent_class)->finalize(budgetp);
|
|
}
|
|
|
|
static void
|
|
gnc_budget_get_property( GObject* object,
|
|
guint prop_id,
|
|
GValue* value,
|
|
GParamSpec* pspec)
|
|
{
|
|
GncBudget* budget;
|
|
GncBudgetPrivate* priv;
|
|
|
|
g_return_if_fail(GNC_IS_BUDGET(object));
|
|
|
|
budget = GNC_BUDGET(object);
|
|
priv = GET_PRIVATE(budget);
|
|
switch ( prop_id )
|
|
{
|
|
case PROP_NAME:
|
|
g_value_set_string(value, priv->name);
|
|
break;
|
|
case PROP_DESCRIPTION:
|
|
g_value_set_string(value, priv->description);
|
|
break;
|
|
case PROP_NUM_PERIODS:
|
|
g_value_set_uint(value, priv->num_periods);
|
|
break;
|
|
case PROP_RECURRENCE:
|
|
/* TODO: Make this a BOXED type */
|
|
g_value_set_pointer(value, &priv->recurrence);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gnc_budget_set_property( GObject* object,
|
|
guint prop_id,
|
|
const GValue* value,
|
|
GParamSpec* pspec)
|
|
{
|
|
GncBudget* budget;
|
|
|
|
g_return_if_fail(GNC_IS_BUDGET(object));
|
|
|
|
budget = GNC_BUDGET(object);
|
|
if (prop_id < PROP_RUNTIME_0)
|
|
g_assert (qof_instance_get_editlevel(budget));
|
|
|
|
switch ( prop_id )
|
|
{
|
|
case PROP_NAME:
|
|
gnc_budget_set_name(budget, g_value_get_string(value));
|
|
break;
|
|
case PROP_DESCRIPTION:
|
|
gnc_budget_set_description(budget, g_value_get_string(value));
|
|
break;
|
|
case PROP_NUM_PERIODS:
|
|
gnc_budget_set_num_periods(budget, g_value_get_uint(value));
|
|
break;
|
|
case PROP_RECURRENCE:
|
|
gnc_budget_set_recurrence(budget, g_value_get_pointer(value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gnc_budget_class_init(GncBudgetClass* klass)
|
|
{
|
|
GObjectClass* gobject_class = G_OBJECT_CLASS(klass);
|
|
|
|
gobject_class->dispose = gnc_budget_dispose;
|
|
gobject_class->finalize = gnc_budget_finalize;
|
|
gobject_class->get_property = gnc_budget_get_property;
|
|
gobject_class->set_property = gnc_budget_set_property;
|
|
|
|
g_object_class_install_property(
|
|
gobject_class,
|
|
PROP_NAME,
|
|
g_param_spec_string( "name",
|
|
"Budget Name",
|
|
"The name is an arbitrary string "
|
|
"assigned by the user. It is intended "
|
|
"to be a short, 5 to 30 character long string "
|
|
"that is displayed by the GUI as the "
|
|
"budget mnemonic",
|
|
NULL,
|
|
G_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property(
|
|
gobject_class,
|
|
PROP_DESCRIPTION,
|
|
g_param_spec_string( "description",
|
|
"Budget Description",
|
|
"The description is an arbitrary string "
|
|
"assigned by the user. It is intended "
|
|
"to be a longer, 1-5 sentence description of "
|
|
"what the budget is all about.",
|
|
NULL,
|
|
G_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property(
|
|
gobject_class,
|
|
PROP_NUM_PERIODS,
|
|
g_param_spec_uint( "num-periods",
|
|
"Number of Periods",
|
|
"The number of periods for this budget.",
|
|
0,
|
|
G_MAXUINT32,
|
|
12,
|
|
G_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property(
|
|
gobject_class,
|
|
PROP_RECURRENCE,
|
|
g_param_spec_pointer( "recurrence",
|
|
"Budget Recurrence",
|
|
"about.",
|
|
G_PARAM_READWRITE));
|
|
}
|
|
|
|
static void commit_err (QofInstance *inst, QofBackendError errcode)
|
|
{
|
|
PERR ("Failed to commit: %d", errcode);
|
|
gnc_engine_signal_commit_error( errcode );
|
|
}
|
|
|
|
static void
|
|
gnc_budget_free(QofInstance *inst)
|
|
{
|
|
GncBudget *budget;
|
|
GncBudgetPrivate* priv;
|
|
|
|
if (inst == NULL)
|
|
return;
|
|
g_return_if_fail(GNC_IS_BUDGET(inst));
|
|
|
|
budget = GNC_BUDGET(inst);
|
|
priv = GET_PRIVATE(budget);
|
|
|
|
/* We first send the message that this object is about to be
|
|
* destroyed so that any GUI elements can remove it before it is
|
|
* actually gone. */
|
|
qof_event_gen( &budget->inst, QOF_EVENT_DESTROY, NULL);
|
|
|
|
CACHE_REMOVE(priv->name);
|
|
CACHE_REMOVE(priv->description);
|
|
|
|
/* qof_instance_release (&budget->inst); */
|
|
g_object_unref(budget);
|
|
}
|
|
|
|
static void noop (QofInstance *inst) {}
|
|
|
|
void
|
|
gnc_budget_begin_edit(GncBudget *bgt)
|
|
{
|
|
qof_begin_edit(QOF_INSTANCE(bgt));
|
|
}
|
|
|
|
void
|
|
gnc_budget_commit_edit(GncBudget *bgt)
|
|
{
|
|
if (!qof_commit_edit(QOF_INSTANCE(bgt))) return;
|
|
qof_commit_edit_part2(QOF_INSTANCE(bgt), commit_err,
|
|
noop, gnc_budget_free);
|
|
}
|
|
|
|
GncBudget*
|
|
gnc_budget_new(QofBook *book)
|
|
{
|
|
GncBudget* budget;
|
|
|
|
g_return_val_if_fail(book, NULL);
|
|
|
|
ENTER(" ");
|
|
budget = g_object_new(GNC_TYPE_BUDGET, NULL);
|
|
qof_instance_init_data (&budget->inst, GNC_ID_BUDGET, book);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_CREATE , NULL);
|
|
|
|
LEAVE(" ");
|
|
return budget;
|
|
}
|
|
|
|
void
|
|
gnc_budget_destroy(GncBudget *budget)
|
|
{
|
|
g_return_if_fail(GNC_IS_BUDGET(budget));
|
|
gnc_budget_begin_edit(budget);
|
|
qof_instance_set_dirty(&budget->inst);
|
|
qof_instance_set_destroying(budget, TRUE);
|
|
gnc_budget_commit_edit(budget);
|
|
}
|
|
|
|
/** Data structure for containing info while cloning budget values */
|
|
typedef struct
|
|
{
|
|
const GncBudget* old_b;
|
|
GncBudget* new_b;
|
|
guint num_periods;
|
|
} CloneBudgetData_t;
|
|
|
|
static void
|
|
clone_budget_values_cb(Account* a, gpointer user_data)
|
|
{
|
|
CloneBudgetData_t* data = (CloneBudgetData_t*)user_data;
|
|
guint i;
|
|
|
|
for ( i = 0; i < data->num_periods; ++i )
|
|
{
|
|
if ( gnc_budget_is_account_period_value_set(data->old_b, a, i) )
|
|
{
|
|
gnc_budget_set_account_period_value(data->new_b, a, i,
|
|
gnc_budget_get_account_period_value(data->old_b, a, i));
|
|
}
|
|
}
|
|
}
|
|
|
|
GncBudget*
|
|
gnc_budget_clone(const GncBudget* old_b)
|
|
{
|
|
GncBudget* new_b;
|
|
Account* root;
|
|
CloneBudgetData_t clone_data;
|
|
|
|
g_return_val_if_fail(old_b != NULL, NULL);
|
|
|
|
ENTER(" ");
|
|
|
|
new_b = gnc_budget_new(qof_instance_get_book(old_b));
|
|
gnc_budget_begin_edit(new_b);
|
|
gnc_budget_set_name(new_b, gnc_budget_get_name(old_b));
|
|
gnc_budget_set_description(new_b, gnc_budget_get_description(old_b));
|
|
gnc_budget_set_recurrence(new_b, gnc_budget_get_recurrence(old_b));
|
|
gnc_budget_set_num_periods(new_b, gnc_budget_get_num_periods(old_b));
|
|
|
|
root = gnc_book_get_root_account(qof_instance_get_book(old_b));
|
|
clone_data.old_b = old_b;
|
|
clone_data.new_b = new_b;
|
|
clone_data.num_periods = gnc_budget_get_num_periods(new_b);
|
|
gnc_account_foreach_descendant(root, clone_budget_values_cb, &clone_data);
|
|
|
|
gnc_budget_commit_edit(new_b);
|
|
|
|
LEAVE(" ");
|
|
|
|
return new_b;
|
|
}
|
|
|
|
void
|
|
gnc_budget_set_name(GncBudget* budget, const gchar* name)
|
|
{
|
|
GncBudgetPrivate* priv;
|
|
|
|
g_return_if_fail(GNC_IS_BUDGET(budget) && name);
|
|
|
|
priv = GET_PRIVATE(budget);
|
|
if ( name == priv->name ) return;
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
CACHE_REPLACE(priv->name, name);
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
}
|
|
|
|
const gchar*
|
|
gnc_budget_get_name(const GncBudget* budget)
|
|
{
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), NULL);
|
|
return GET_PRIVATE(budget)->name;
|
|
}
|
|
|
|
void
|
|
gnc_budget_set_description(GncBudget* budget, const gchar* description)
|
|
{
|
|
GncBudgetPrivate* priv;
|
|
|
|
g_return_if_fail(GNC_IS_BUDGET(budget));
|
|
g_return_if_fail(description);
|
|
|
|
priv = GET_PRIVATE(budget);
|
|
if ( description == priv->description ) return;
|
|
gnc_budget_begin_edit(budget);
|
|
CACHE_REPLACE(priv->description, description);
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
}
|
|
|
|
const gchar*
|
|
gnc_budget_get_description(const GncBudget* budget)
|
|
{
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), NULL);
|
|
return GET_PRIVATE(budget)->description;
|
|
}
|
|
|
|
void
|
|
gnc_budget_set_recurrence(GncBudget *budget, const Recurrence *r)
|
|
{
|
|
GncBudgetPrivate* priv;
|
|
|
|
g_return_if_fail(budget && r);
|
|
priv = GET_PRIVATE(budget);
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
priv->recurrence = *r;
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen(&budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
}
|
|
|
|
const Recurrence *
|
|
gnc_budget_get_recurrence(const GncBudget *budget)
|
|
{
|
|
g_return_val_if_fail(budget, NULL);
|
|
return (&GET_PRIVATE(budget)->recurrence);
|
|
}
|
|
|
|
const GncGUID*
|
|
gnc_budget_get_guid(const GncBudget* budget)
|
|
{
|
|
g_return_val_if_fail(budget, NULL);
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), NULL);
|
|
return qof_instance_get_guid(QOF_INSTANCE(budget));
|
|
}
|
|
|
|
void
|
|
gnc_budget_set_num_periods(GncBudget* budget, guint num_periods)
|
|
{
|
|
GncBudgetPrivate* priv;
|
|
|
|
g_return_if_fail(GNC_IS_BUDGET(budget));
|
|
|
|
priv = GET_PRIVATE(budget);
|
|
if ( priv->num_periods == num_periods ) return;
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
priv->num_periods = num_periods;
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
}
|
|
|
|
guint
|
|
gnc_budget_get_num_periods(const GncBudget* budget)
|
|
{
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), 0);
|
|
return GET_PRIVATE(budget)->num_periods;
|
|
}
|
|
|
|
static inline void
|
|
make_period_path (const Account *account, guint period_num, char *path1, char *path2)
|
|
{
|
|
const GncGUID *guid;
|
|
gchar *bufend;
|
|
guid = xaccAccountGetGUID (account);
|
|
guid_to_string_buff (guid, path1);
|
|
g_sprintf (path2, "%d", period_num);
|
|
}
|
|
|
|
/* period_num is zero-based */
|
|
/* What happens when account is deleted, after we have an entry for it? */
|
|
void
|
|
gnc_budget_unset_account_period_value(GncBudget *budget, const Account *account,
|
|
guint period_num)
|
|
{
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
|
|
g_return_if_fail (budget != NULL);
|
|
g_return_if_fail (account != NULL);
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
qof_instance_set_kvp (QOF_INSTANCE (budget), NULL, 2, path_part_one, path_part_two);
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
|
|
}
|
|
|
|
/* period_num is zero-based */
|
|
/* What happens when account is deleted, after we have an entry for it? */
|
|
void
|
|
gnc_budget_set_account_period_value(GncBudget *budget, const Account *account,
|
|
guint period_num, gnc_numeric val)
|
|
{
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
|
|
/* Watch out for an off-by-one error here:
|
|
* period_num starts from 0 while num_periods starts from 1 */
|
|
if (period_num >= GET_PRIVATE(budget)->num_periods)
|
|
{
|
|
PWARN("Period %i does not exist", period_num);
|
|
return;
|
|
}
|
|
|
|
g_return_if_fail (budget != NULL);
|
|
g_return_if_fail (account != NULL);
|
|
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
if (gnc_numeric_check(val))
|
|
qof_instance_set_kvp (QOF_INSTANCE (budget), NULL, 2, path_part_one, path_part_two);
|
|
else
|
|
{
|
|
GValue v = G_VALUE_INIT;
|
|
g_value_init (&v, GNC_TYPE_NUMERIC);
|
|
g_value_set_boxed (&v, &val);
|
|
qof_instance_set_kvp (QOF_INSTANCE (budget), &v, 2, path_part_one, path_part_two);
|
|
g_value_unset (&v);
|
|
}
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
|
|
}
|
|
|
|
/* We don't need these here, but maybe they're useful somewhere else?
|
|
Maybe this should move to Account.h */
|
|
|
|
gboolean
|
|
gnc_budget_is_account_period_value_set(const GncBudget *budget,
|
|
const Account *account,
|
|
guint period_num)
|
|
{
|
|
GValue v = G_VALUE_INIT;
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
gconstpointer ptr = NULL;
|
|
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), FALSE);
|
|
g_return_val_if_fail(account, FALSE);
|
|
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
qof_instance_get_kvp (QOF_INSTANCE (budget), &v, 2, path_part_one, path_part_two);
|
|
if (G_VALUE_HOLDS_BOXED (&v))
|
|
ptr = g_value_get_boxed (&v);
|
|
g_value_unset (&v);
|
|
return (ptr != NULL);
|
|
}
|
|
|
|
gnc_numeric
|
|
gnc_budget_get_account_period_value(const GncBudget *budget,
|
|
const Account *account,
|
|
guint period_num)
|
|
{
|
|
gnc_numeric *numeric = NULL;
|
|
gnc_numeric retval;
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
GValue v = G_VALUE_INIT;
|
|
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), gnc_numeric_zero());
|
|
g_return_val_if_fail(account, gnc_numeric_zero());
|
|
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
qof_instance_get_kvp (QOF_INSTANCE (budget), &v, 2, path_part_one, path_part_two);
|
|
if (G_VALUE_HOLDS_BOXED (&v))
|
|
numeric = (gnc_numeric*)g_value_get_boxed (&v);
|
|
|
|
retval = numeric ? *numeric : gnc_numeric_zero ();
|
|
g_value_unset (&v);
|
|
return retval;
|
|
}
|
|
|
|
|
|
void
|
|
gnc_budget_set_account_period_note(GncBudget *budget, const Account *account,
|
|
guint period_num, const gchar *note)
|
|
{
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
|
|
/* Watch out for an off-by-one error here:
|
|
* period_num starts from 0 while num_periods starts from 1 */
|
|
if (period_num >= GET_PRIVATE(budget)->num_periods)
|
|
{
|
|
PWARN("Period %i does not exist", period_num);
|
|
return;
|
|
}
|
|
|
|
g_return_if_fail (budget != NULL);
|
|
g_return_if_fail (account != NULL);
|
|
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
|
|
gnc_budget_begin_edit(budget);
|
|
if (note == NULL)
|
|
qof_instance_set_kvp (QOF_INSTANCE (budget), NULL, 3, GNC_BUDGET_NOTES_PATH, path_part_one, path_part_two);
|
|
else
|
|
{
|
|
GValue v = G_VALUE_INIT;
|
|
g_value_init (&v, G_TYPE_STRING);
|
|
g_value_set_string (&v, note);
|
|
|
|
qof_instance_set_kvp (QOF_INSTANCE (budget), &v, 3, GNC_BUDGET_NOTES_PATH, path_part_one, path_part_two);
|
|
g_value_unset (&v);
|
|
}
|
|
qof_instance_set_dirty(&budget->inst);
|
|
gnc_budget_commit_edit(budget);
|
|
|
|
qof_event_gen( &budget->inst, QOF_EVENT_MODIFY, NULL);
|
|
|
|
}
|
|
|
|
gchar *
|
|
gnc_budget_get_account_period_note(const GncBudget *budget,
|
|
const Account *account, guint period_num)
|
|
{
|
|
gchar path_part_one [GUID_ENCODING_LENGTH + 1];
|
|
gchar path_part_two [GNC_BUDGET_MAX_NUM_PERIODS_DIGITS];
|
|
GValue v = G_VALUE_INIT;
|
|
gchar *retval;
|
|
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget), NULL);
|
|
g_return_val_if_fail(account, NULL);
|
|
|
|
make_period_path (account, period_num, path_part_one, path_part_two);
|
|
qof_instance_get_kvp (QOF_INSTANCE (budget), &v, 3, GNC_BUDGET_NOTES_PATH, path_part_one, path_part_two);
|
|
retval = G_VALUE_HOLDS_STRING (&v) ? g_value_dup_string(&v) : NULL;
|
|
g_value_unset (&v);
|
|
return retval;
|
|
}
|
|
|
|
time64
|
|
gnc_budget_get_period_start_date(const GncBudget *budget, guint period_num)
|
|
{
|
|
g_return_val_if_fail (GNC_IS_BUDGET(budget), 0);
|
|
return recurrenceGetPeriodTime(&GET_PRIVATE(budget)->recurrence, period_num, FALSE);
|
|
}
|
|
|
|
time64
|
|
gnc_budget_get_period_end_date(const GncBudget *budget, guint period_num)
|
|
{
|
|
g_return_val_if_fail (GNC_IS_BUDGET(budget), 0);
|
|
return recurrenceGetPeriodTime(&GET_PRIVATE(budget)->recurrence, period_num, TRUE);
|
|
}
|
|
|
|
gnc_numeric
|
|
gnc_budget_get_account_period_actual_value(
|
|
const GncBudget *budget, Account *acc, guint period_num)
|
|
{
|
|
// FIXME: maybe zero is not best error return val.
|
|
g_return_val_if_fail(GNC_IS_BUDGET(budget) && acc, gnc_numeric_zero());
|
|
return recurrenceGetAccountPeriodValue(&GET_PRIVATE(budget)->recurrence,
|
|
acc, period_num);
|
|
}
|
|
|
|
GncBudget*
|
|
gnc_budget_lookup (const GncGUID *guid, const QofBook *book)
|
|
{
|
|
QofCollection *col;
|
|
|
|
g_return_val_if_fail(guid, NULL);
|
|
g_return_val_if_fail(book, NULL);
|
|
col = qof_book_get_collection (book, GNC_ID_BUDGET);
|
|
return GNC_BUDGET(qof_collection_lookup_entity (col, guid));
|
|
}
|
|
|
|
static void just_get_one(QofInstance *ent, gpointer data)
|
|
{
|
|
GncBudget **bgt = (GncBudget**)data;
|
|
if (bgt && !*bgt) *bgt = GNC_BUDGET(ent);
|
|
}
|
|
|
|
GncBudget*
|
|
gnc_budget_get_default (QofBook *book)
|
|
{
|
|
QofCollection *col;
|
|
GncBudget *bgt = NULL;
|
|
GncGUID *default_budget_guid = NULL;
|
|
|
|
g_return_val_if_fail(book, NULL);
|
|
|
|
qof_instance_get (QOF_INSTANCE (book),
|
|
"default-budget", &default_budget_guid,
|
|
NULL);
|
|
if (default_budget_guid)
|
|
{
|
|
col = qof_book_get_collection(book, GNC_ID_BUDGET);
|
|
bgt = (GncBudget *) qof_collection_lookup_entity(col,
|
|
default_budget_guid);
|
|
}
|
|
|
|
/* Revert to 2.2.x behavior if the book has no default budget. */
|
|
|
|
if ( bgt == NULL )
|
|
{
|
|
col = qof_book_get_collection(book, GNC_ID_BUDGET);
|
|
if (qof_collection_count(col) > 0)
|
|
{
|
|
qof_collection_foreach(col, just_get_one, &bgt);
|
|
}
|
|
}
|
|
|
|
guid_free (default_budget_guid);
|
|
return bgt;
|
|
}
|
|
|
|
static void
|
|
destroy_budget_on_book_close(QofInstance *ent, gpointer data)
|
|
{
|
|
GncBudget* bgt = GNC_BUDGET(ent);
|
|
|
|
gnc_budget_destroy(bgt);
|
|
}
|
|
|
|
/** Handles book end - frees all budgets from the book
|
|
*
|
|
* @param book Book being closed
|
|
*/
|
|
static void
|
|
gnc_budget_book_end(QofBook* book)
|
|
{
|
|
QofCollection *col;
|
|
|
|
col = qof_book_get_collection(book, GNC_ID_BUDGET);
|
|
qof_collection_foreach(col, destroy_budget_on_book_close, NULL);
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
/* MSVC compiler doesn't have C99 "designated initializers"
|
|
* so we wrap them in a macro that is empty on MSVC. */
|
|
# define DI(x) /* */
|
|
#else
|
|
# define DI(x) x
|
|
#endif
|
|
|
|
/* Define the QofObject. */
|
|
static QofObject budget_object_def =
|
|
{
|
|
DI(.interface_version = ) QOF_OBJECT_VERSION,
|
|
DI(.e_type = ) GNC_ID_BUDGET,
|
|
DI(.type_label = ) "Budget",
|
|
DI(.create = ) (gpointer)gnc_budget_new,
|
|
DI(.book_begin = ) NULL,
|
|
DI(.book_end = ) gnc_budget_book_end,
|
|
DI(.is_dirty = ) qof_collection_is_dirty,
|
|
DI(.mark_clean = ) qof_collection_mark_clean,
|
|
DI(.foreach = ) qof_collection_foreach,
|
|
DI(.printable = ) (const char * (*)(gpointer)) gnc_budget_get_name,
|
|
DI(.version_cmp = ) (int (*)(gpointer, gpointer)) qof_instance_version_cmp,
|
|
};
|
|
|
|
|
|
/* Static wrapper getters for the recurrence params */
|
|
static PeriodType gnc_budget_get_rec_pt(const GncBudget *bgt)
|
|
{
|
|
return recurrenceGetPeriodType(&(GET_PRIVATE(bgt)->recurrence));
|
|
}
|
|
static guint gnc_budget_get_rec_mult(const GncBudget *bgt)
|
|
{
|
|
return recurrenceGetMultiplier(&(GET_PRIVATE(bgt)->recurrence));
|
|
}
|
|
static time64 gnc_budget_get_rec_time(const GncBudget *bgt)
|
|
{
|
|
return recurrenceGetTime(&(GET_PRIVATE(bgt)->recurrence));
|
|
}
|
|
|
|
/* Register ourselves with the engine. */
|
|
gboolean gnc_budget_register (void)
|
|
{
|
|
static QofParam params[] =
|
|
{
|
|
{
|
|
"name", QOF_TYPE_STRING,
|
|
(QofAccessFunc) gnc_budget_get_name,
|
|
(QofSetterFunc) gnc_budget_set_name
|
|
},
|
|
{
|
|
"description", QOF_TYPE_STRING,
|
|
(QofAccessFunc) gnc_budget_get_description,
|
|
(QofSetterFunc) gnc_budget_set_description
|
|
},
|
|
{
|
|
"recurrence_period_type", QOF_TYPE_INT32,
|
|
(QofAccessFunc) gnc_budget_get_rec_pt, NULL
|
|
},
|
|
/* Signedness problem: Should be unsigned. */
|
|
{
|
|
"recurrence_multiplier", QOF_TYPE_INT32,
|
|
(QofAccessFunc) gnc_budget_get_rec_mult, NULL
|
|
},
|
|
/* This is the same way that SchedXaction.c uses QOF_TYPE_DATE
|
|
but I don't think QOF actually supports a GDate, so I think
|
|
this is wrong. */
|
|
{
|
|
"recurrence_date", QOF_TYPE_DATE,
|
|
(QofAccessFunc) gnc_budget_get_rec_time, NULL
|
|
},
|
|
/* Signedness problem: Should be unsigned. */
|
|
{
|
|
"num_periods", QOF_TYPE_INT32,
|
|
(QofAccessFunc) gnc_budget_get_num_periods,
|
|
(QofSetterFunc) gnc_budget_set_num_periods
|
|
},
|
|
{
|
|
QOF_PARAM_BOOK, QOF_ID_BOOK,
|
|
(QofAccessFunc) qof_instance_get_book, NULL
|
|
},
|
|
{
|
|
QOF_PARAM_GUID, QOF_TYPE_GUID,
|
|
(QofAccessFunc) qof_instance_get_guid, NULL
|
|
},
|
|
{ NULL },
|
|
};
|
|
|
|
qof_class_register(GNC_ID_BUDGET, (QofSortFunc) NULL, params);
|
|
return qof_object_register(&budget_object_def);
|
|
}
|