gnucash/src/business/business-core/gncBillTerm.c

664 lines
14 KiB
C
Raw Normal View History

/*
* gncBillTerm.c -- the Gnucash Billing Terms interface
* Copyright (C) 2002 Derek Atkins
* Author: Derek Atkins <warlord@MIT.EDU>
*/
#include "config.h"
#include <glib.h>
#include "messages.h"
#include "gnc-book.h"
#include "gnc-numeric.h"
#include "gnc-engine-util.h"
#include "GNCIdP.h"
#include "QueryCore.h"
#include "QueryNew.h"
#include "QueryObject.h"
#include "gnc-event-p.h"
#include "gnc-be-utils.h"
#include "gncBusiness.h"
#include "gncBillTermP.h"
struct _gncBillTerm {
GUID guid;
char * name;
char * desc;
GncBillTermType type;
gint due_days;
gint disc_days;
gnc_numeric discount;
gint cutoff;
gint64 refcount;
GNCBook * book;
GncBillTerm * parent; /* if non-null, we are an immutable child */
GncBillTerm * child; /* if non-null, we have not changed */
gboolean invisible;
GList * children; /* list of children for disconnection */
int editlevel;
gboolean do_free;
gboolean dirty;
};
struct _book_info {
GncBookInfo bi;
GList * terms; /* visible terms */
};
static short module = MOD_BUSINESS;
#define _GNC_MOD_NAME GNC_BILLTERM_MODULE_NAME
#define CACHE_INSERT(str) g_cache_insert(gnc_engine_get_string_cache(), (gpointer)(str));
#define CACHE_REMOVE(str) g_cache_remove(gnc_engine_get_string_cache(), (str));
#define SET_STR(obj, member, str) { \
char * tmp; \
\
if (!safe_strcmp (member, str)) return; \
gncBillTermBeginEdit (obj); \
tmp = CACHE_INSERT (str); \
CACHE_REMOVE (member); \
member = tmp; \
}
static void add_or_rem_object (GncBillTerm *term, gboolean add);
static void addObj (GncBillTerm *term);
static void remObj (GncBillTerm *term);
static void maybe_resort_list (GncBillTerm *term);
G_INLINE_FUNC void mark_term (GncBillTerm *term);
G_INLINE_FUNC void
mark_term (GncBillTerm *term)
{
term->dirty = TRUE;
gncBusinessSetDirtyFlag (term->book, _GNC_MOD_NAME, TRUE);
gnc_engine_generate_event (&term->guid, GNC_EVENT_MODIFY);
}
/* Create/Destroy Functions */
GncBillTerm * gncBillTermCreate (GNCBook *book)
{
GncBillTerm *term;
if (!book) return NULL;
term = g_new0 (GncBillTerm, 1);
term->book = book;
term->name = CACHE_INSERT ("");
term->desc = CACHE_INSERT ("");
term->discount = gnc_numeric_zero ();
xaccGUIDNew (&term->guid, book);
addObj (term);
gnc_engine_generate_event (&term->guid, GNC_EVENT_CREATE);
return term;
}
void gncBillTermDestroy (GncBillTerm *term)
{
if (!term) return;
term->do_free = TRUE;
gncBusinessSetDirtyFlag (term->book, _GNC_MOD_NAME, TRUE);
gncBillTermCommitEdit (term);
}
static void gncBillTermFree (GncBillTerm *term)
{
GncBillTerm *child;
GList *list;
if (!term) return;
gnc_engine_generate_event (&term->guid, GNC_EVENT_DESTROY);
CACHE_REMOVE (term->name);
CACHE_REMOVE (term->desc);
remObj (term);
if (!term->do_free)
PERR("free a billterm without do_free set!");
/* disconnect from the children */
for (list = term->children; list; list=list->next) {
child = list->data;
gncBillTermSetParent(child, NULL);
}
g_list_free(term->children);
g_free (term);
}
static void
gncBillTermAddChild (GncBillTerm *table, GncBillTerm *child)
{
g_return_if_fail(table);
g_return_if_fail(child);
g_return_if_fail(table->do_free == FALSE);
table->children = g_list_prepend(table->children, child);
}
static void
gncBillTermRemoveChild (GncBillTerm *table, GncBillTerm *child)
{
g_return_if_fail(table);
g_return_if_fail(child);
if (table->do_free)
return;
table->children = g_list_remove(table->children, child);
}
/* Set Functions */
void gncBillTermSetGUID (GncBillTerm *term, const GUID *guid)
{
if (!term || !guid) return;
if (guid_equal (guid, &term->guid)) return;
gncBillTermBeginEdit (term);
remObj (term);
term->guid = *guid;
addObj (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetName (GncBillTerm *term, const char *name)
{
if (!term || !name) return;
SET_STR (term, term->name, name);
mark_term (term);
maybe_resort_list (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetDescription (GncBillTerm *term, const char *desc)
{
if (!term || !desc) return;
SET_STR (term, term->desc, desc);
mark_term (term);
maybe_resort_list (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetType (GncBillTerm *term, GncBillTermType type)
{
if (!term) return;
if (term->type == type) return;
gncBillTermBeginEdit (term);
term->type = type;
mark_term (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetDueDays (GncBillTerm *term, gint days)
{
if (!term) return;
if (term->due_days == days) return;
gncBillTermBeginEdit (term);
term->due_days = days;
mark_term (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetDiscountDays (GncBillTerm *term, gint days)
{
if (!term) return;
if (term->disc_days == days) return;
gncBillTermBeginEdit (term);
term->disc_days = days;
mark_term (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetDiscount (GncBillTerm *term, gnc_numeric discount)
{
if (!term) return;
if (gnc_numeric_eq (term->discount, discount)) return;
gncBillTermBeginEdit (term);
term->discount = discount;
mark_term (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetCutoff (GncBillTerm *term, gint cutoff)
{
if (!term) return;
if (term->cutoff == cutoff) return;
gncBillTermBeginEdit (term);
term->cutoff = cutoff;
mark_term (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetParent (GncBillTerm *term, GncBillTerm *parent)
{
if (!term) return;
gncBillTermBeginEdit (term);
if (term->parent)
gncBillTermRemoveChild(term->parent, term);
term->parent = parent;
if (parent)
gncBillTermAddChild(parent, term);
term->refcount = 0;
gncBillTermMakeInvisible (term);
gncBillTermCommitEdit (term);
}
void gncBillTermSetChild (GncBillTerm *term, GncBillTerm *child)
{
if (!term) return;
gncBillTermBeginEdit (term);
term->child = child;
gncBillTermCommitEdit (term);
}
void gncBillTermIncRef (GncBillTerm *term)
{
if (!term) return;
if (term->parent || term->invisible) return; /* children dont need refcounts */
gncBillTermBeginEdit (term);
term->refcount++;
gncBillTermCommitEdit (term);
}
void gncBillTermDecRef (GncBillTerm *term)
{
if (!term) return;
if (term->parent || term->invisible) return; /* children dont need refcounts */
gncBillTermBeginEdit (term);
term->refcount--;
g_return_if_fail (term->refcount >= 0);
gncBillTermCommitEdit (term);
}
void gncBillTermSetRefcount (GncBillTerm *term, gint64 refcount)
{
if (!term) return;
term->refcount = refcount;
}
void gncBillTermMakeInvisible (GncBillTerm *term)
{
if (!term) return;
gncBillTermBeginEdit (term);
term->invisible = TRUE;
add_or_rem_object (term, FALSE);
gncBillTermCommitEdit (term);
}
void gncBillTermChanged (GncBillTerm *term)
{
if (!term) return;
term->child = NULL;
}
void gncBillTermBeginEdit (GncBillTerm *term)
{
GNC_BEGIN_EDIT (term, _GNC_MOD_NAME);
}
static void gncBillTermOnError (GncBillTerm *term, GNCBackendError errcode)
{
PERR("BillTerm Backend Failure: %d", errcode);
}
static void gncBillTermOnDone (GncBillTerm *term)
{
term->dirty = FALSE;
}
void gncBillTermCommitEdit (GncBillTerm *term)
{
GNC_COMMIT_EDIT_PART1 (term);
GNC_COMMIT_EDIT_PART2 (term, _GNC_MOD_NAME, gncBillTermOnError,
gncBillTermOnDone, gncBillTermFree);
}
/* Get Functions */
GncBillTerm * gncBillTermLookup (GNCBook *book, const GUID *guid)
{
if (!book || !guid) return NULL;
return xaccLookupEntity (gnc_book_get_entity_table (book),
guid, _GNC_MOD_NAME);
}
GncBillTerm *gncBillTermLookupByName (GNCBook *book, const char *name)
{
GList *list = gncBillTermGetTerms (book);
for ( ; list; list = list->next) {
GncBillTerm *term = list->data;
if (!safe_strcmp (term->name, name))
return list->data;
}
return NULL;
}
GList * gncBillTermGetTerms (GNCBook *book)
{
struct _book_info *bi;
if (!book) return NULL;
bi = gnc_book_get_data (book, _GNC_MOD_NAME);
return bi->terms;
}
const GUID *gncBillTermGetGUID (GncBillTerm *term)
{
if (!term) return NULL;
return &term->guid;
}
GNCBook *gncBillTermGetBook (GncBillTerm *term)
{
if (!term) return NULL;
return term->book;
}
const char *gncBillTermGetName (GncBillTerm *term)
{
if (!term) return NULL;
return term->name;
}
const char *gncBillTermGetDescription (GncBillTerm *term)
{
if (!term) return NULL;
return term->desc;
}
GncBillTermType gncBillTermGetType (GncBillTerm *term)
{
if (!term) return 0;
return term->type;
}
gint gncBillTermGetDueDays (GncBillTerm *term)
{
if (!term) return 0;
return term->due_days;
}
gint gncBillTermGetDiscountDays (GncBillTerm *term)
{
if (!term) return 0;
return term->disc_days;
}
gnc_numeric gncBillTermGetDiscount (GncBillTerm *term)
{
if (!term) return gnc_numeric_zero ();
return term->discount;
}
gint gncBillTermGetCutoff (GncBillTerm *term)
{
if (!term) return 0;
return term->cutoff;
}
static GncBillTerm *gncBillTermCopy (GncBillTerm *term)
{
GncBillTerm *t;
if (!term) return NULL;
t = gncBillTermCreate (term->book);
gncBillTermBeginEdit(t);
gncBillTermSetName (t, term->name);
gncBillTermSetDescription (t, term->desc);
t->type = term->type;
t->due_days = term->due_days;
t->disc_days = term->disc_days;
t->discount = term->discount;
t->cutoff = term->cutoff;
gncBillTermCommitEdit(t);
return t;
}
GncBillTerm *gncBillTermReturnChild (GncBillTerm *term, gboolean make_new)
{
GncBillTerm *child = NULL;
if (!term) return NULL;
if (term->child) return term->child;
if (term->parent || term->invisible) return term;
if (make_new) {
child = gncBillTermCopy (term);
gncBillTermSetChild (term, child);
gncBillTermSetParent (child, term);
}
return child;
}
GncBillTerm *gncBillTermGetParent (GncBillTerm *term)
{
if (!term) return NULL;
return term->parent;
}
gint64 gncBillTermGetRefcount (GncBillTerm *term)
{
if (!term) return 0;
return term->refcount;
}
gboolean gncBillTermGetInvisible (GncBillTerm *term)
{
if (!term) return FALSE;
return term->invisible;
}
int gncBillTermCompare (GncBillTerm *a, GncBillTerm *b)
{
int ret;
if (!a && !b) return 0;
if (!a) return -1;
if (!b) return 1;
ret = safe_strcmp (a->name, b->name);
if (ret) return ret;
return safe_strcmp (a->desc, b->desc);
}
gboolean gncBillTermIsDirty (GncBillTerm *term)
{
if (!term) return FALSE;
return term->dirty;
}
/********************************************************/
/* functions to compute dates from Bill Terms */
#define SECS_PER_DAY 86400
/* Based on the timespec and a proximo type, compute the month and
* year this is due. The actual day is filled in below.
*/
static void
compute_monthyear (GncBillTerm *term, Timespec post_date,
int *month, int *year)
{
int iday, imonth, iyear;
int cutoff = term->cutoff;
g_return_if_fail (term->type == GNC_TERM_TYPE_PROXIMO);
gnc_timespec2dmy (post_date, &iday, &imonth, &iyear);
if (cutoff <= 0)
cutoff += gnc_timespec_last_mday (post_date);
if (iday <= cutoff) {
/* We apply this to next month */
imonth++;
} else {
/* We apply to the following month */
imonth += 2;
}
if (imonth > 12) {
iyear++;
imonth -= 12;
}
if (month) *month = imonth;
if (year) *year = iyear;
}
static Timespec
compute_time (GncBillTerm *term, Timespec post_date, int days)
{
Timespec res = post_date;
int day, month, year;
switch (term->type) {
case GNC_TERM_TYPE_DAYS:
res.tv_sec += (SECS_PER_DAY * days);
break;
case GNC_TERM_TYPE_PROXIMO:
compute_monthyear (term, post_date, &month, &year);
day = gnc_date_my_last_mday (month, year);
if (days < day)
day = days;
res = gnc_dmy2timespec (day, month, year);
break;
}
return res;
}
Timespec
gncBillTermComputeDueDate (GncBillTerm *term, Timespec post_date)
{
Timespec res = post_date;
if (!term) return res;
return compute_time (term, post_date, term->due_days);
}
Timespec
gncBillTermComputeDiscountDate (GncBillTerm *term, Timespec post_date)
{
Timespec res = post_date;
if (!term) return res;
return compute_time (term, post_date, term->disc_days);
}
/* Package-Private functions */
static void maybe_resort_list (GncBillTerm *term)
{
struct _book_info *bi;
if (term->parent || term->invisible) return;
bi = gnc_book_get_data (term->book, _GNC_MOD_NAME);
bi->terms = g_list_sort (bi->terms, (GCompareFunc)gncBillTermCompare);
}
static void add_or_rem_object (GncBillTerm *term, gboolean add)
{
struct _book_info *bi;
if (!term) return;
bi = gnc_book_get_data (term->book, _GNC_MOD_NAME);
if (add)
bi->terms = g_list_insert_sorted (bi->terms, term,
(GCompareFunc)gncBillTermCompare);
else
bi->terms = g_list_remove (bi->terms, term);
}
static void addObj (GncBillTerm *term)
{
gncBusinessAddObject (term->book, _GNC_MOD_NAME, term, &term->guid);
add_or_rem_object (term, TRUE);
}
static void remObj (GncBillTerm *term)
{
gncBusinessRemoveObject (term->book, _GNC_MOD_NAME, &term->guid);
add_or_rem_object (term, FALSE);
}
static void _gncBillTermCreate (GNCBook *book)
{
struct _book_info *bi;
if (!book) return;
bi = g_new0 (struct _book_info, 1);
bi->bi.ht = guid_hash_table_new ();
gnc_book_set_data (book, _GNC_MOD_NAME, bi);
}
static void _gncBillTermDestroy (GNCBook *book)
{
struct _book_info *bi;
if (!book) return;
bi = gnc_book_get_data (book, _GNC_MOD_NAME);
/* XXX : Destroy the objects? */
g_hash_table_destroy (bi->bi.ht);
g_list_free (bi->terms);
g_free (bi);
}
static gboolean _gncBillTermIsDirty (GNCBook *book)
{
return gncBusinessIsDirty (book, _GNC_MOD_NAME);
}
static void _gncBillTermMarkClean (GNCBook *book)
{
gncBusinessSetDirtyFlag (book, _GNC_MOD_NAME, FALSE);
}
static void _gncBillTermForeach (GNCBook *book, foreachObjectCB cb,
gpointer user_data)
{
gncBusinessForeach (book, _GNC_MOD_NAME, cb, user_data);
}
static GncObject_t gncBillTermDesc = {
GNC_OBJECT_VERSION,
_GNC_MOD_NAME,
"Billing Term",
_gncBillTermCreate,
_gncBillTermDestroy,
_gncBillTermIsDirty,
_gncBillTermMarkClean,
_gncBillTermForeach,
NULL /* printable */
};
gboolean gncBillTermRegister (void)
{
static QueryObjectDef params[] = {
{ QUERY_PARAM_BOOK, GNC_ID_BOOK, (QueryAccess)gncBillTermGetBook },
{ QUERY_PARAM_GUID, QUERYCORE_GUID, (QueryAccess)gncBillTermGetGUID },
{ NULL },
};
gncQueryObjectRegister (_GNC_MOD_NAME, (QuerySort)gncBillTermCompare, params);
return gncObjectRegister (&gncBillTermDesc);
}