gnucash unstable: Multiple changes pushed

John Ralls jralls at code.gnucash.org
Sun Mar 25 12:55:50 EDT 2018


Updated	 via  https://github.com/Gnucash/gnucash/commit/dedb5265 (commit)
	 via  https://github.com/Gnucash/gnucash/commit/a0f49d8b (commit)
	 via  https://github.com/Gnucash/gnucash/commit/9cd22ee6 (commit)
	 via  https://github.com/Gnucash/gnucash/commit/a670238f (commit)
	 via  https://github.com/Gnucash/gnucash/commit/2b6047f9 (commit)
	from  https://github.com/Gnucash/gnucash/commit/8081e78e (commit)



commit dedb5265f0317bc9ebb99b1ca73279c7963d4b7e
Merge: a0f49d8 9cd22ee
Author: John Ralls <jralls at ceridwen.us>
Date:   Sun Mar 25 09:36:21 2018 -0700

    Merge branch 'maint' into unstable

diff --cc gnucash/gnome/gnc-plugin-page-account-tree.c
index 176e9bc,0000000..5913b75
mode 100644,000000..100644
--- a/gnucash/gnome/gnc-plugin-page-account-tree.c
+++ b/gnucash/gnome/gnc-plugin-page-account-tree.c
@@@ -1,1808 -1,0 +1,1808 @@@
 +/*
 + * gnc-plugin-page-account-tree.c --
 + *
 + * Copyright (C) 2003 Jan Arne Petersen <jpetersen at uni-bonn.de>
 + * Copyright (C) 2003,2005,2006 David Hampton <hampton at 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 at gnu.org
 + */
 +
 +/** @addtogroup ContentPlugins
 +    @{ */
 +/** @addtogroup GncPluginPageAccountTree An Account Tree Plugin
 +    @{ */
 +/** @file gnc-plugin-page-account-tree.c
 +    @brief Functions providing a chart of account page.
 +    @author Copyright (C) 2003 Jan Arne Petersen <jpetersen at uni-bonn.de>
 +    @author Copyright (C) 2003,2005,2006 David Hampton <hampton at employees.org>
 +*/
 +
 +#include <config.h>
 +
 +#include <gtk/gtk.h>
 +#include <glib/gi18n.h>
 +#include "gnc-plugin-page-account-tree.h"
 +#include "gnc-plugin-page-register.h"
 +#include "gnc-plugin-page-register2.h"
 +
 +#include "Scrub.h"
 +#include "Scrub3.h"
 +#include "ScrubBusiness.h"
 +#include "Transaction.h"
 +#include "dialog-account.h"
 +#include "dialog-transfer.h"
 +#include "dialog-utils.h"
 +#include "assistant-hierarchy.h"
 +#include "gnc-account-sel.h"
 +#include "gnc-component-manager.h"
 +#include "gnc-engine.h"
 +#include "gnc-gnome-utils.h"
 +#include "gnc-gobject-utils.h"
 +#include "gnc-icons.h"
 +#include "gnc-plugin-account-tree.h"
 +#include "gnc-prefs.h"
 +#include "gnc-session.h"
 +#include "gnc-split-reg.h"
 +#include "gnc-state.h"
 +#include "gnc-tree-view-account.h"
 +#include "gnc-tree-model-account-types.h"
 +#include "gnc-ui.h"
 +#include "gnc-ui-util.h"
 +#include "gnc-window.h"
 +#include "dialog-lot-viewer.h"
 +#include "window-reconcile.h"
 +#include "window-autoclear.h"
 +#include "window-main-summarybar.h"
 +#include "dialog-object-references.h"
 +#include "dialog-find-account.h"
 +
 +/* This static indicates the debugging module that this .o belongs to.  */
 +static QofLogModule log_module = GNC_MOD_GUI;
 +
 +#define PLUGIN_PAGE_ACCT_TREE_CM_CLASS "plugin-page-acct-tree"
 +#define STATE_SECTION "Account Hierarchy"
 +
 +#define DELETE_DIALOG_FILTER       "filter"
 +#define DELETE_DIALOG_ACCOUNT      "account"
 +#define DELETE_DIALOG_TRANS_MAS    "trans_mas"
 +#define DELETE_DIALOG_SA_MAS       "sa_mas"
 +#define DELETE_DIALOG_SA_TRANS_MAS "sa_trans_mas"
 +#define DELETE_DIALOG_SA_TRANS     "sa_trans"
 +#define DELETE_DIALOG_SA_SPLITS    "sa_has_split"
 +#define DELETE_DIALOG_OK_BUTTON    "deletebutton"
 +
 +enum
 +{
 +    ACCOUNT_SELECTED,
 +    LAST_SIGNAL
 +};
 +
 +typedef struct GncPluginPageAccountTreePrivate
 +{
 +    GtkWidget   *widget;
 +    GtkTreeView *tree_view;
 +    gint         component_id;
 +    AccountFilterDialog fd;
 +} GncPluginPageAccountTreePrivate;
 +
 +#define GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(o)  \
 +   (G_TYPE_INSTANCE_GET_PRIVATE ((o), GNC_TYPE_PLUGIN_PAGE_ACCOUNT_TREE, GncPluginPageAccountTreePrivate))
 +
 +static GObjectClass *parent_class = NULL;
 +
 +/************************************************************
 + *                        Prototypes                        *
 + ************************************************************/
 +/* Plugin Actions */
 +static void gnc_plugin_page_account_tree_class_init (GncPluginPageAccountTreeClass *klass);
 +static void gnc_plugin_page_account_tree_init (GncPluginPageAccountTree *plugin_page);
 +static void gnc_plugin_page_account_tree_finalize (GObject *object);
 +static void gnc_plugin_page_account_tree_selected (GObject *object, gpointer user_data);
 +
 +static GtkWidget *gnc_plugin_page_account_tree_create_widget (GncPluginPage *plugin_page);
 +static void gnc_plugin_page_account_tree_destroy_widget (GncPluginPage *plugin_page);
 +static void gnc_plugin_page_account_tree_save_page (GncPluginPage *plugin_page, GKeyFile *file, const gchar *group);
 +static GncPluginPage *gnc_plugin_page_account_tree_recreate_page (GtkWidget *window, GKeyFile *file, const gchar *group);
 +
 +/* Callbacks */
 +static void gnc_plugin_page_account_tree_summarybar_position_changed(gpointer prefs, gchar* pref, gpointer user_data);
 +static gboolean gnc_plugin_page_account_tree_button_press_cb (GtkWidget *widget,
 +        GdkEventButton *event,
 +        GncPluginPage *page);
 +static void gnc_plugin_page_account_tree_double_click_cb (GtkTreeView        *treeview,
 +        GtkTreePath        *path,
 +        GtkTreeViewColumn  *col,
 +        GncPluginPageAccountTree *page);
 +
 +static void gnc_plugin_page_account_tree_selection_changed_cb (GtkTreeSelection *selection,
 +        GncPluginPageAccountTree *page);
 +void gppat_populate_trans_mas_list(GtkToggleButton *sa_mrb, GtkWidget *dialog);
 +void gppat_set_insensitive_iff_rb_active(GtkWidget *widget, GtkToggleButton *b);
 +
 +/* Command callbacks */
 +static void gnc_plugin_page_account_tree_cmd_new_account (GtkAction *action, GncPluginPageAccountTree *plugin_page);
 +static void gnc_plugin_page_account_tree_cmd_file_new_hierarchy (GtkAction *action, GncPluginPageAccountTree *plugin_page);
 +static void gnc_plugin_page_account_tree_cmd_open_account (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_open_subaccounts (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_edit_account (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_find_account (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_find_account_popup (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_delete_account (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_renumber_accounts (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_view_filter_by (GtkAction *action, GncPluginPageAccountTree *plugin_page);
 +static void gnc_plugin_page_account_tree_cmd_reconcile (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_autoclear (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_transfer (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_stock_split (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_lots (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_scrub (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_scrub_sub (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_scrub_all (GtkAction *action, GncPluginPageAccountTree *page);
 +
 +/* Command callback for new Register Test */
 +static void gnc_plugin_page_account_tree_cmd_open2_account (GtkAction *action, GncPluginPageAccountTree *page);
 +static void gnc_plugin_page_account_tree_cmd_open2_subaccounts (GtkAction *action, GncPluginPageAccountTree *page);
 +
 +static guint plugin_page_signals[LAST_SIGNAL] = { 0 };
 +
 +
 +static GtkActionEntry gnc_plugin_page_account_tree_actions [] =
 +{
 +    /* Toplevel */
 +    { "FakeToplevel", NULL, "", NULL, NULL, NULL },
 +
 +    /* File menu */
 +    {
 +        "FileNewAccountAction", GNC_ICON_NEW_ACCOUNT, N_("New _Account..."), NULL,
 +        N_("Create a new Account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_new_account)
 +    },
 +    {
 +        "FileAddAccountHierarchyAssistantAction", GNC_ICON_NEW_ACCOUNT, N_("New Account _Hierarchy..."), NULL,
 +        N_("Extend the current book by merging with new account type categories"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_file_new_hierarchy)
 +    },
 +#ifdef REGISTER2_ENABLED
 +    {
 +        "FileOpenAccount2Action", GNC_ICON_OPEN_ACCOUNT, N_("Open _Account"), NULL,
 +        N_("Open the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open2_account)
 +    },
 +    {
 +        "FileOpenAccountAction", GNC_ICON_OPEN_ACCOUNT, N_("Open _Old Style Register Account"), NULL,
 +        N_("Open the old style register selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open_account)
 +    },
 +#else
 +    {
 +        "FileOpenAccountAction", GNC_ICON_OPEN_ACCOUNT, N_("Open _Account"), NULL,
 +        N_("Open the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open_account)
 +    },
 +#endif
 +
 +#ifdef REGISTER2_ENABLED
 +    {
 +        "FileOpenSubaccounts2Action", GNC_ICON_OPEN_ACCOUNT, N_("Open _SubAccounts"), NULL,
 +        N_("Open the selected account and all its subaccounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open2_subaccounts)
 +    },
 +    {
 +        "FileOpenSubaccountsAction", GNC_ICON_OPEN_ACCOUNT, N_("Open Old St_yle Subaccounts"), NULL,
 +        N_("Open the old style register selected account and all its subaccounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open_subaccounts)
 +    },
 +#else
 +    {
 +        "FileOpenSubaccountsAction", GNC_ICON_OPEN_ACCOUNT, N_("Open _SubAccounts"), NULL,
 +        N_("Open the selected account and all its subaccounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open_subaccounts)
 +    },
 +#endif
 +
 +    /* Edit menu */
 +    {
 +        "EditEditAccountAction", GNC_ICON_EDIT_ACCOUNT, N_("Edit _Account"), "<primary>e",
 +        N_("Edit the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_edit_account)
 +    },
 +    {
 +        "EditDeleteAccountAction", GNC_ICON_DELETE_ACCOUNT, N_("_Delete Account..."), "Delete",
 +        N_("Delete selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_delete_account)
 +    },
 +    {
 +        "EditFindAccountAction", "edit-find", N_("F_ind Account"), "<primary>i",
 +        N_("Find an account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_find_account)
 +    },
 +    {
 +        "EditFindAccountPopupAction", "edit-find", N_("F_ind Account"), "<primary>i",
 +        N_("Find an account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_find_account_popup)
 +    },
 +    {
 +        "EditRenumberSubaccountsAction", NULL, N_("_Renumber Subaccounts..."), NULL,
 +        N_("Renumber the children of the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_renumber_accounts)
 +    },
 +
 +    /* View menu */
 +    {
 +        "ViewFilterByAction", NULL, N_("_Filter By..."), NULL, NULL,
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_view_filter_by)
 +    },
 +
 +    /* Actions menu */
 +    {
 +        "ActionsReconcileAction", NULL, N_("_Reconcile..."), NULL,
 +        N_("Reconcile the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_reconcile)
 +    },
 +    {
 +        "ActionsAutoClearAction", NULL, N_("_Auto-clear..."), NULL,
 +        N_("Automatically clear individual transactions, given a cleared amount"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_autoclear)
 +    },
 +    {
 +        "ActionsTransferAction", NULL, N_("_Transfer..."), "<primary>t",
 +        N_("Transfer funds from one account to another"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_transfer)
 +    },
 +    {
 +        "ActionsStockSplitAction", NULL, N_("Stoc_k Split..."), NULL,
 +        N_("Record a stock split or a stock merger"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_stock_split)
 +    },
 +    {
 +        "ActionsLotsAction", NULL, N_("View _Lots..."), NULL,
 +        N_("Bring up the lot viewer/editor window"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_lots)
 +    },
 +    {
 +        "ScrubAction", NULL, N_("Check & Repair A_ccount"), NULL,
 +        N_("Check for and repair unbalanced transactions and orphan splits " "in this account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_scrub)
 +    },
 +    {
 +        "ScrubSubAction", NULL, N_("Check & Repair Su_baccounts"), NULL,
 +        N_("Check for and repair unbalanced transactions and orphan splits "
 +        "in this account and its subaccounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_scrub_sub)
 +    },
 +    {
 +        "ScrubAllAction", NULL, N_("Check & Repair A_ll"), NULL,
 +        N_("Check for and repair unbalanced transactions and orphan splits " "in all accounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_scrub_all)
 +    },
 +    /* Extensions Menu */
 +    { "Register2TestAction", NULL, N_("_Register2"), NULL, NULL, NULL },
 +    {
 +        "Register2TestAccountAction", GNC_ICON_OPEN_ACCOUNT, N_("Open _Account"), NULL,
 +        N_("Open the selected account"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open2_account)
 +    },
 +    {
 +        "Register2TestSubAccountAction", GNC_ICON_OPEN_ACCOUNT, N_("Open _SubAccounts"), NULL,
 +        N_("Open the selected account and all its subaccounts"),
 +        G_CALLBACK (gnc_plugin_page_account_tree_cmd_open2_subaccounts)
 +    },
 +};
 +/** The number of actions provided by this plugin. */
 +static guint gnc_plugin_page_account_tree_n_actions = G_N_ELEMENTS (gnc_plugin_page_account_tree_actions);
 +
 +
 +/** Actions that require an account to be selected before they are
 + *  enabled, and the book is in read-write mode. */
 +static const gchar *actions_requiring_account_rw[] =
 +{
 +    "EditEditAccountAction",
 +    "EditDeleteAccountAction",
 +    "ActionsReconcileAction",
 +    "ActionsAutoClearAction",
 +    NULL
 +};
 +
 +/** Actions that require an account to be selected before they are
 + *  enabled. Those actions can be selected even if the book is in readonly mode. */
 +static const gchar *actions_requiring_account_always[] =
 +{
 +    "FileOpenAccountAction",
 +#ifdef REGISTER2_ENABLED
 +    "FileOpenAccount2Action",
 +#endif
 +    "FileOpenSubaccountsAction",
 +    "ActionsLotsAction",
 +    NULL
 +};
 +
 +/* This is the list of actions which are switched inactive in a read-only book. */
 +static const gchar* readonly_inactive_actions[] =
 +{
 +    "FileNewAccountAction",
 +    "FileAddAccountHierarchyAssistantAction",
 +    "EditEditAccountAction",
 +    "EditDeleteAccountAction",
 +    "EditRenumberSubaccountsAction",
 +    "ActionsTransferAction",
 +    "ActionsReconcileAction",
 +    "ActionsAutoClearAction",
 +    "ActionsStockSplitAction",
 +    "ScrubAction",
 +    "ScrubSubAction",
 +    "ScrubAllAction",
 +    NULL
 +};
 +
 +/** Short labels for use on the toolbar buttons. */
 +static action_toolbar_labels toolbar_labels[] =
 +{
 +    { "FileOpenAccountAction", 	            N_("Open") },
 +#ifdef REGISTER2_ENABLED
 +    { "FileOpenAccount2Action", 	    N_("Open2") },
 +#endif
 +    { "EditEditAccountAction", 	            N_("Edit") },
 +    { "FileNewAccountAction",    	    N_("New") },
 +    { "EditDeleteAccountAction", 	    N_("Delete") },
 +    { NULL, NULL },
 +};
 +
 +
 +GType
 +gnc_plugin_page_account_tree_get_type (void)
 +{
 +    static GType gnc_plugin_page_account_tree_type = 0;
 +
 +    if (gnc_plugin_page_account_tree_type == 0)
 +    {
 +        static const GTypeInfo our_info =
 +        {
 +            sizeof (GncPluginPageAccountTreeClass),
 +            NULL,
 +            NULL,
 +            (GClassInitFunc) gnc_plugin_page_account_tree_class_init,
 +            NULL,
 +            NULL,
 +            sizeof (GncPluginPageAccountTree),
 +            0,
 +            (GInstanceInitFunc) gnc_plugin_page_account_tree_init
 +        };
 +
 +        gnc_plugin_page_account_tree_type = g_type_register_static (GNC_TYPE_PLUGIN_PAGE,
 +                                            GNC_PLUGIN_PAGE_ACCOUNT_TREE_NAME,
 +                                            &our_info, 0);
 +    }
 +
 +    return gnc_plugin_page_account_tree_type;
 +}
 +
 +GncPluginPage *
 +gnc_plugin_page_account_tree_new (void)
 +{
 +    GncPluginPageAccountTree *plugin_page;
 +
 +    ENTER(" ");
 +    plugin_page = g_object_new (GNC_TYPE_PLUGIN_PAGE_ACCOUNT_TREE,
 +                                NULL);
 +
 +    LEAVE("new account tree page %p", plugin_page);
 +    return GNC_PLUGIN_PAGE (plugin_page);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_class_init (GncPluginPageAccountTreeClass *klass)
 +{
 +    GObjectClass *object_class = G_OBJECT_CLASS (klass);
 +    GncPluginPageClass *gnc_plugin_class = GNC_PLUGIN_PAGE_CLASS(klass);
 +
 +    parent_class = g_type_class_peek_parent (klass);
 +
 +    object_class->finalize = gnc_plugin_page_account_tree_finalize;
 +
 +    gnc_plugin_class->tab_icon        = GNC_ICON_ACCOUNT;
 +    gnc_plugin_class->plugin_name     = GNC_PLUGIN_PAGE_ACCOUNT_TREE_NAME;
 +    gnc_plugin_class->create_widget   = gnc_plugin_page_account_tree_create_widget;
 +    gnc_plugin_class->destroy_widget  = gnc_plugin_page_account_tree_destroy_widget;
 +    gnc_plugin_class->save_page       = gnc_plugin_page_account_tree_save_page;
 +    gnc_plugin_class->recreate_page   = gnc_plugin_page_account_tree_recreate_page;
 +
 +    g_type_class_add_private(klass, sizeof(GncPluginPageAccountTreePrivate));
 +
 +    plugin_page_signals[ACCOUNT_SELECTED] =
 +        g_signal_new ("account_selected",
 +                      G_OBJECT_CLASS_TYPE (object_class),
 +                      G_SIGNAL_RUN_FIRST,
 +                      G_STRUCT_OFFSET (GncPluginPageAccountTreeClass, account_selected),
 +                      NULL, NULL,
 +                      g_cclosure_marshal_VOID__POINTER,
 +                      G_TYPE_NONE, 1,
 +                      G_TYPE_POINTER);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_init (GncPluginPageAccountTree *plugin_page)
 +{
 +    GtkActionGroup *action_group;
 +    GncPluginPageAccountTreePrivate *priv;
 +    GncPluginPage *parent;
 +    const GList *page_list;
 +
 +    ENTER("page %p", plugin_page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(plugin_page);
 +
 +    /* Init parent declared variables */
 +    parent = GNC_PLUGIN_PAGE(plugin_page);
 +#ifdef REGISTER2_ENABLED
 +    g_object_set(G_OBJECT(plugin_page),
 +                 "page-name",      _("Accounts"),
 +                 "page-uri",       "default:",
 +                 "ui-description", "gnc-plugin-page-account-tree2-ui.xml",
 +                 NULL);
 +#else
 +    g_object_set(G_OBJECT(plugin_page),
 +                 "page-name",      _("Accounts"),
 +                 "page-uri",       "default:",
 +                 "ui-description", "gnc-plugin-page-account-tree-ui.xml",
 +                 NULL);
 +#endif
 +    g_signal_connect (G_OBJECT (plugin_page), "selected",
 +                      G_CALLBACK (gnc_plugin_page_account_tree_selected), plugin_page);
 +
 +    /* change me when the system supports multiple books */
 +    gnc_plugin_page_add_book(parent, gnc_get_current_book());
 +
 +    /* Is this the first accounts page? */
 +    page_list =
 +        gnc_gobject_tracking_get_list(GNC_PLUGIN_PAGE_ACCOUNT_TREE_NAME);
 +    if (plugin_page == page_list->data)
 +    {
 +        g_object_set_data(G_OBJECT(plugin_page), PLUGIN_PAGE_IMMUTABLE,
 +                          GINT_TO_POINTER(1));
 +    }
 +
 +    /* Create menu and toolbar information */
 +    action_group =
 +        gnc_plugin_page_create_action_group(parent,
 +                                            "GncPluginPageAccountTreeActions");
 +    gtk_action_group_add_actions(action_group,
 +                                 gnc_plugin_page_account_tree_actions,
 +                                 gnc_plugin_page_account_tree_n_actions,
 +                                 plugin_page);
 +    gnc_plugin_init_short_names (action_group, toolbar_labels);
 +
 +    /* Visible types */
 +    priv->fd.visible_types = -1; /* Start with all types */
 +    priv->fd.show_hidden = FALSE;
 +    priv->fd.show_unused = TRUE;
 +    priv->fd.show_zero_total = TRUE;
 +    priv->fd.filter_override = g_hash_table_new (g_direct_hash, g_direct_equal);
 +
 +    LEAVE("page %p, priv %p, action group %p",
 +          plugin_page, priv, action_group);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_finalize (GObject *object)
 +{
 +    GncPluginPageAccountTree *page;
 +    GncPluginPageAccountTreePrivate *priv;
 +
 +    ENTER("object %p", object);
 +    page = GNC_PLUGIN_PAGE_ACCOUNT_TREE (object);
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +    g_return_if_fail (priv != NULL);
 +
 +    G_OBJECT_CLASS (parent_class)->finalize (object);
 +    LEAVE(" ");
 +}
 +
 +void
 +gnc_plugin_page_account_tree_open (Account *account, GtkWindow *win)
 +{
 +    GncPluginPageAccountTreePrivate *priv;
 +    GncPluginPageAccountTree *page;
 +    GncPluginPage *plugin_page = NULL;
 +    const GList *page_list;
 +    GtkWidget   *window;
 +
 +    /* Find Accounts page */
 +    page_list = gnc_gobject_tracking_get_list(GNC_PLUGIN_PAGE_ACCOUNT_TREE_NAME);
 +
 +    // If we have a window, look for account page in that window
 +    if (g_list_length ((GList*)page_list) != 0)
 +    {
 +        if (win != NULL)
 +        {
 +            for ( ; page_list; page_list = g_list_next(page_list))
 +            {
 +                plugin_page = GNC_PLUGIN_PAGE(page_list->data);
 +                if (GTK_WINDOW(plugin_page->window) == win)
 +                    break;
 +            }
 +        }
 +        else // if no window, open first account page in list
 +            plugin_page = GNC_PLUGIN_PAGE(page_list->data);
 +    }
 +    else // we have no account pages, create one
 +        plugin_page = gnc_plugin_page_account_tree_new ();
 +
 +    window = plugin_page->window;
 +
 +    gnc_main_window_open_page (GNC_MAIN_WINDOW(window), plugin_page);
 +
 +    page = GNC_PLUGIN_PAGE_ACCOUNT_TREE (plugin_page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +
 +    if (account != NULL)
 +    {
 +        Account *root_account = gnc_get_current_root_account ();
 +        Account *parent_account = NULL;
 +        Account *temp_account = account;
 +
 +        g_hash_table_insert (priv->fd.filter_override, account, account);
 +
 +        // make sure we override all the parent accounts to root
 +        while (parent_account != root_account)
 +        {
 +            parent_account = gnc_account_get_parent (temp_account);
 +
 +            g_hash_table_insert (priv->fd.filter_override, parent_account, parent_account);
 +            temp_account = parent_account;
 +        }
 +        gnc_tree_view_account_refilter (GNC_TREE_VIEW_ACCOUNT(priv->tree_view));
 +        gnc_tree_view_account_set_selected_account (GNC_TREE_VIEW_ACCOUNT(priv->tree_view), account);
 +    }
 +}
 +
 +Account *
 +gnc_plugin_page_account_tree_get_current_account (GncPluginPageAccountTree *page)
 +{
 +    GncPluginPageAccountTreePrivate *priv;
 +    Account *account;
 +
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +    ENTER("page %p (tree view %p)", page, priv->tree_view);
 +    account = gnc_tree_view_account_get_selected_account (GNC_TREE_VIEW_ACCOUNT(priv->tree_view));
 +    if (account == NULL)
 +    {
 +        LEAVE("no account");
 +        return NULL;
 +    }
 +
 +    LEAVE("account %p", account);
 +    return account;
 +}
 +
 +gboolean
 +gnc_plugin_page_account_tree_focus (GncPluginPageAccountTree *page)
 +{
 +    if (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE(page))
 +    {
 +        GncPluginPageAccountTreePrivate *priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +        GtkTreeView *view = GTK_TREE_VIEW(priv->tree_view);
 +
 +        if (!gtk_widget_is_focus (GTK_WIDGET(view)))
 +            gtk_widget_grab_focus (GTK_WIDGET(view));
 +    }
 +    return FALSE;
 +}
 +
 +/* Virtual Functions */
 +
 +static void
 +gnc_plugin_page_account_refresh_cb (GHashTable *changes, gpointer user_data)
 +{
 +    GncPluginPageAccountTree *page = user_data;
 +    GncPluginPageAccountTreePrivate *priv;
 +
 +    g_return_if_fail(GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE(page));
 +
 +    /* We're only looking for forced updates here. */
 +    if (changes)
 +        return;
 +
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +    gtk_widget_queue_draw(priv->widget);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_close_cb (gpointer user_data)
 +{
 +    GncPluginPage *plugin_page = GNC_PLUGIN_PAGE(user_data);
 +    gnc_main_window_close_page(plugin_page);
 +}
 +
 +static void
 +gnc_plugin_page_account_editing_started_cd (gpointer various, GncPluginPageRegister *page)
 +{
 +    GncPluginPage *plugin_page = GNC_PLUGIN_PAGE(page);
 +    GtkAction *action = gnc_main_window_find_action (GNC_MAIN_WINDOW(plugin_page->window),
 +                                                     "EditDeleteAccountAction");
 +
 +    if (action != NULL)
 +        gtk_action_set_sensitive (action, FALSE);
 +}
 +
 +static void
 +gnc_plugin_page_account_editing_finished_cb (gpointer various, GncPluginPageRegister *page)
 +{
 +    GncPluginPage *plugin_page = GNC_PLUGIN_PAGE(page);
 +    GtkAction *action = gnc_main_window_find_action (GNC_MAIN_WINDOW(plugin_page->window),
 +                                                     "EditDeleteAccountAction");
 +
 +    if (action != NULL)
 +        gtk_action_set_sensitive (action, TRUE);
 +}
 +
 +static GtkWidget *
 +gnc_plugin_page_account_tree_create_widget (GncPluginPage *plugin_page)
 +{
 +    GncPluginPageAccountTree *page;
 +    GncPluginPageAccountTreePrivate *priv;
 +    GtkTreeSelection *selection;
 +    GtkTreeView *tree_view;
 +    GtkWidget *scrolled_window;
 +    GtkTreeViewColumn *col;
 +
 +    ENTER("page %p", plugin_page);
 +    page = GNC_PLUGIN_PAGE_ACCOUNT_TREE (plugin_page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +    if (priv->widget != NULL)
 +    {
 +        LEAVE("widget = %p", priv->widget);
 +        return priv->widget;
 +    }
 +
 +    priv->widget = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
 +    gtk_box_set_homogeneous (GTK_BOX (priv->widget), FALSE);
 +    gtk_widget_show (priv->widget);
 +
 +    // Set the style context for this page so it can be easily manipulated with css
 +    gnc_widget_set_style_context (GTK_WIDGET(priv->widget), "GncAccountPage");
 +
 +    scrolled_window = gtk_scrolled_window_new (NULL, NULL);
 +    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window),
 +                                    GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
 +    gtk_widget_show (scrolled_window);
 +    gtk_box_pack_start (GTK_BOX (priv->widget), scrolled_window,
 +                        TRUE, TRUE, 0);
 +
 +    tree_view = gnc_tree_view_account_new(FALSE);
 +    col = gnc_tree_view_find_column_by_name(
 +              GNC_TREE_VIEW(tree_view), "description");
 +    g_object_set_data(G_OBJECT(col), DEFAULT_VISIBLE, GINT_TO_POINTER(1));
 +    col = gnc_tree_view_find_column_by_name(
 +              GNC_TREE_VIEW(tree_view), "total");
 +    g_object_set_data(G_OBJECT(col), DEFAULT_VISIBLE, GINT_TO_POINTER(1));
 +    gnc_tree_view_configure_columns(GNC_TREE_VIEW(tree_view));
 +    g_object_set(G_OBJECT(tree_view),
 +                 "state-section", STATE_SECTION,
 +                 "show-column-menu", TRUE,
 +                 NULL);
 +
 +    /* No name handler; then the user can't click on the name of the
 +       account to open its register. */
 +    gnc_tree_view_account_set_code_edited(GNC_TREE_VIEW_ACCOUNT(tree_view),
 +                                          gnc_tree_view_account_code_edited_cb);
 +    gnc_tree_view_account_set_description_edited(GNC_TREE_VIEW_ACCOUNT(tree_view),
 +            gnc_tree_view_account_description_edited_cb);
 +    gnc_tree_view_account_set_notes_edited(GNC_TREE_VIEW_ACCOUNT(tree_view),
 +                                           gnc_tree_view_account_notes_edited_cb);
 +
 +    // Setup some callbacks so menu actions can be disabled/enabled
 +    gnc_tree_view_account_set_editing_started_cb(GNC_TREE_VIEW_ACCOUNT(tree_view),
 +        (GFunc)gnc_plugin_page_account_editing_started_cd, page);
 +    gnc_tree_view_account_set_editing_finished_cb(GNC_TREE_VIEW_ACCOUNT(tree_view),
 +        (GFunc)gnc_plugin_page_account_editing_finished_cb, page);
 +
 +    priv->tree_view = tree_view;
 +    selection = gtk_tree_view_get_selection(tree_view);
 +    g_signal_connect (G_OBJECT (selection), "changed",
 +                      G_CALLBACK (gnc_plugin_page_account_tree_selection_changed_cb), page);
 +    g_signal_connect (G_OBJECT (tree_view), "button-press-event",
 +                      G_CALLBACK (gnc_plugin_page_account_tree_button_press_cb), page);
 +    g_signal_connect (G_OBJECT (tree_view), "row-activated",
 +                      G_CALLBACK (gnc_plugin_page_account_tree_double_click_cb), page);
 +
 +    gtk_tree_view_set_headers_visible(tree_view, TRUE);
 +    gnc_plugin_page_account_tree_selection_changed_cb (NULL, page);
 +    gtk_widget_show (GTK_WIDGET (tree_view));
 +    gtk_container_add (GTK_CONTAINER (scrolled_window), GTK_WIDGET(tree_view));
 +
 +    priv->fd.tree_view = GNC_TREE_VIEW_ACCOUNT(priv->tree_view);
 +    gnc_tree_view_account_set_filter (
 +        GNC_TREE_VIEW_ACCOUNT(tree_view),
 +        gnc_plugin_page_account_tree_filter_accounts, &priv->fd, NULL);
 +
 +    priv->component_id =
 +        gnc_register_gui_component(PLUGIN_PAGE_ACCT_TREE_CM_CLASS,
 +                                   gnc_plugin_page_account_refresh_cb,
 +                                   gnc_plugin_page_account_tree_close_cb,
 +                                   page);
 +    gnc_gui_component_set_session (priv->component_id,
 +                                   gnc_get_current_session());
 +
 +    plugin_page->summarybar = gnc_main_window_summary_new();
 +    gtk_box_pack_start (GTK_BOX (priv->widget), plugin_page->summarybar,
 +                        FALSE, FALSE, 0);
 +    gtk_widget_show(plugin_page->summarybar);
 +    gnc_plugin_page_account_tree_summarybar_position_changed(NULL, NULL, page);
 +    gnc_prefs_register_cb (GNC_PREFS_GROUP_GENERAL,
 +                           GNC_PREF_SUMMARYBAR_POSITION_TOP,
 +                           gnc_plugin_page_account_tree_summarybar_position_changed,
 +                           page);
 +    gnc_prefs_register_cb (GNC_PREFS_GROUP_GENERAL,
 +                           GNC_PREF_SUMMARYBAR_POSITION_BOTTOM,
 +                           gnc_plugin_page_account_tree_summarybar_position_changed,
 +                           page);
 +
 +    LEAVE("widget = %p", priv->widget);
 +    return priv->widget;
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_destroy_widget (GncPluginPage *plugin_page)
 +{
 +    GncPluginPageAccountTree *page;
 +    GncPluginPageAccountTreePrivate *priv;
 +
 +    ENTER("page %p", plugin_page);
 +    page = GNC_PLUGIN_PAGE_ACCOUNT_TREE (plugin_page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +
 +    gnc_prefs_remove_cb_by_func (GNC_PREFS_GROUP_GENERAL,
 +                                 GNC_PREF_SUMMARYBAR_POSITION_TOP,
 +                                 gnc_plugin_page_account_tree_summarybar_position_changed,
 +                                 page);
 +    gnc_prefs_remove_cb_by_func (GNC_PREFS_GROUP_GENERAL,
 +                                 GNC_PREF_SUMMARYBAR_POSITION_BOTTOM,
 +                                 gnc_plugin_page_account_tree_summarybar_position_changed,
 +                                 page);
 +
 +    // Destroy the filter override hash table
 +    g_hash_table_destroy(priv->fd.filter_override);
 +
 +    // Remove the page focus idle function if present
 +    g_idle_remove_by_data (GNC_PLUGIN_PAGE_ACCOUNT_TREE (plugin_page));
 +
 +    if (priv->widget)
 +    {
 +        g_object_unref(G_OBJECT(priv->widget));
 +        priv->widget = NULL;
 +    }
 +
 +    if (priv->component_id)
 +    {
 +        gnc_unregister_gui_component(priv->component_id);
 +        priv->component_id = 0;
 +    }
 +
 +    LEAVE("widget destroyed");
 +}
 +
 +static void update_inactive_actions(GncPluginPage *plugin_page)
 +{
 +    GtkActionGroup *action_group;
 +    gboolean is_sensitive = !qof_book_is_readonly(gnc_get_current_book());
 +
 +    // We are readonly - so we have to switch particular actions to inactive.
 +    g_return_if_fail(plugin_page);
 +    g_return_if_fail(GNC_IS_PLUGIN_PAGE(plugin_page));
 +
 +    /* Get the action group */
 +    action_group = gnc_plugin_page_get_action_group(plugin_page);
 +    g_return_if_fail(GTK_IS_ACTION_GROUP (action_group));
 +
 +    /* Set the action's sensitivity */
 +    gnc_plugin_update_actions (action_group, readonly_inactive_actions,
 +                               "sensitive", is_sensitive);
 +}
 +
 +/**
 + * Called when this page is selected.
 + *
 + * Update the toolbar button sensitivity. */
 +static void gnc_plugin_page_account_tree_selected (GObject *object, gpointer user_data)
 +{
 +    GncPluginPage *plugin_page = GNC_PLUGIN_PAGE (object);
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE (plugin_page));
 +    update_inactive_actions(plugin_page);
 +}
 +
 +/** Save enough information about this account tree page that it can
 + *  be recreated next time the user starts gnucash.
 + *
 + *  @param plugin_page The page to save.
 + *
 + *  @param key_file A pointer to the GKeyFile data structure where the
 + *  page information should be written.
 + *
 + *  @param group_name The group name to use when saving data. */
 +static void
 +gnc_plugin_page_account_tree_save_page (GncPluginPage *plugin_page,
 +                                        GKeyFile *key_file,
 +                                        const gchar *group_name)
 +{
 +    GncPluginPageAccountTree *account_page;
 +    GncPluginPageAccountTreePrivate *priv;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE(plugin_page));
 +    g_return_if_fail (key_file != NULL);
 +    g_return_if_fail (group_name != NULL);
 +
 +    ENTER("page %p, key_file %p, group_name %s", plugin_page, key_file,
 +          group_name);
 +
 +    account_page = GNC_PLUGIN_PAGE_ACCOUNT_TREE(plugin_page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(account_page);
 +
 +    gnc_tree_view_account_save(GNC_TREE_VIEW_ACCOUNT(priv->tree_view),
 +                               &priv->fd, key_file, group_name);
 +    LEAVE(" ");
 +}
 +
 +
 +
 +/** Create a new account tree page based on the information saved
 + *  during a previous instantiation of gnucash.
 + *
 + *  @param window The window where this page should be installed.
 + *
 + *  @param key_file A pointer to the GKeyFile data structure where the
 + *  page information should be read.
 + *
 + *  @param group_name The group name to use when restoring data. */
 +static GncPluginPage *
 +gnc_plugin_page_account_tree_recreate_page (GtkWidget *window,
 +        GKeyFile *key_file,
 +        const gchar *group_name)
 +{
 +    GncPluginPageAccountTree *account_page;
 +    GncPluginPageAccountTreePrivate *priv;
 +    GncPluginPage *page;
 +
 +    g_return_val_if_fail(key_file, NULL);
 +    g_return_val_if_fail(group_name, NULL);
 +    ENTER("key_file %p, group_name %s", key_file, group_name);
 +
 +    /* Create the new page. */
 +    page = gnc_plugin_page_account_tree_new();
 +    account_page = GNC_PLUGIN_PAGE_ACCOUNT_TREE(page);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(account_page);
 +
 +    /* Install it now so we can then manipulate the created widget */
 +    gnc_main_window_open_page(GNC_MAIN_WINDOW(window), page);
 +
 +    gnc_tree_view_account_restore(GNC_TREE_VIEW_ACCOUNT(priv->tree_view),
 +                                  &priv->fd, key_file, group_name);
 +    LEAVE(" ");
 +    return page;
 +}
 +
 +
 +/* Callbacks */
 +
 +static void
 +gnc_plugin_page_account_tree_summarybar_position_changed(gpointer prefs, gchar* pref, gpointer user_data)
 +{
 +    GncPluginPage *plugin_page;
 +    GncPluginPageAccountTree *page;
 +    GncPluginPageAccountTreePrivate *priv;
 +    GtkPositionType position = GTK_POS_BOTTOM;
 +
 +    g_return_if_fail(user_data != NULL);
 +
 +    plugin_page = GNC_PLUGIN_PAGE(user_data);
 +    page = GNC_PLUGIN_PAGE_ACCOUNT_TREE (user_data);
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +
 +    if (gnc_prefs_get_bool (GNC_PREFS_GROUP_GENERAL, GNC_PREF_SUMMARYBAR_POSITION_TOP))
 +        position = GTK_POS_TOP;
 +
 +    gtk_box_reorder_child(GTK_BOX(priv->widget),
 +                          plugin_page->summarybar,
 +                          (position == GTK_POS_TOP ? 0 : -1) );
 +}
 +
 +/** This button press handler calls the common button press handler
 + *  for all pages.  The GtkTreeView eats all button presses and
 + *  doesn't pass them up the widget tree, even when doesn't do
 + *  anything with them.  The only way to get access to the button
 + *  presses in an account tree page is here on the tree view widget.
 + *  Button presses on all other pages are caught by the signal
 + *  registered in gnc-main-window.c. */
 +static gboolean
 +gnc_plugin_page_account_tree_button_press_cb (GtkWidget *widget,
 +        GdkEventButton *event,
 +        GncPluginPage *page)
 +{
 +
 +    g_return_val_if_fail(GNC_IS_PLUGIN_PAGE(page), FALSE);
 +
 +    ENTER("widget %p, event %p, page %p", widget, event, page);
 +    gnc_main_window_button_press_cb(widget, event, page);
 +    LEAVE(" ");
 +
 +    /* Always return FALSE.  This will let the tree view callback run as
 +     * well which will select the item under the cursor.  By the time
 +     * the user sees the menu both callbacks will have run and the menu
 +     * actions will operate on the just-selected account. */
 +    return FALSE;
 +}
 +
 +static void
 +gppat_open_account_common (GncPluginPageAccountTree *page,
 +                           Account *account,
 +                           gboolean include_subs)
 +{
 +    GtkWidget *window;
 +    GncPluginPage *new_page;
 +
 +    if (account == NULL)
 +        return;
 +
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    new_page = gnc_plugin_page_register_new (account, include_subs);
 +    gnc_main_window_open_page (GNC_MAIN_WINDOW(window), new_page);
 +}
 +
 +/*################## Added for Reg2 #################*/
 +/*        New Register Common                        */
 +static void
 +gppat_open2_account_common (GncPluginPageAccountTree *page,
 +                            Account *account,
 +                            gboolean include_subs)
 +{
 +    GtkWidget *window;
 +    GncPluginPage *new_page;
 +
 +    if (account == NULL)
 +        return;
 +
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    new_page = gnc_plugin_page_register2_new (account, include_subs);
 +    gnc_main_window_open_page (GNC_MAIN_WINDOW(window), new_page);
 +}
 +/*################## Added for Reg2 #################*/
 +
 +static void
 +gnc_plugin_page_account_tree_double_click_cb (GtkTreeView *treeview,
 +        GtkTreePath        *path,
 +        GtkTreeViewColumn  *col,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkTreeModel *model;
 +    GtkTreeIter iter;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    g_return_if_fail (treeview);
 +
 +    model = gtk_tree_view_get_model(treeview);
 +    if (gtk_tree_model_get_iter(model, &iter, path))
 +    {
 +        Account *account = gnc_tree_view_account_get_account_from_path (GNC_TREE_VIEW_ACCOUNT(treeview), path);
 +        if (xaccAccountGetPlaceholder (account))
 +        {
 +            /* This is a placeholder account. Only only show/hide
 +             * subaccount list if there is one.
 +             */
 +            if (gtk_tree_model_iter_has_child(model, &iter))
 +            {
 +                /* There are children,
 +                 * just expand or collapse the row. */
 +                if (gtk_tree_view_row_expanded(treeview, path))
 +                    gtk_tree_view_collapse_row(treeview, path);
 +                else
 +                    gtk_tree_view_expand_row(treeview, path, FALSE);
 +            }
 +        }
 +        else
 +        {
 +            /* No placeholder account, so open its register */
 +#ifdef REGISTER2_ENABLED
 +            gppat_open2_account_common (page, account, FALSE);
 +#else
 +            gppat_open_account_common (page, account, FALSE);
 +#endif
 +        }
 +    }
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_selection_changed_cb (GtkTreeSelection *selection,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkActionGroup *action_group;
 +    GtkAction *action;
 +    GtkTreeView *view;
 +    Account *account = NULL;
 +    gboolean sensitive;
 +    gboolean subaccounts;
 +    gboolean is_readwrite = !qof_book_is_readonly(gnc_get_current_book());
 +
 +    g_return_if_fail(GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE(page));
 +
 +    if (!selection)
 +    {
 +        sensitive = FALSE;
 +        subaccounts = FALSE;
 +    }
 +    else
 +    {
 +        g_return_if_fail(GTK_IS_TREE_SELECTION(selection));
 +        view = gtk_tree_selection_get_tree_view (selection);
 +        account = gnc_tree_view_account_get_selected_account (GNC_TREE_VIEW_ACCOUNT(view));
 +        sensitive = (account != NULL);
 +
 +        subaccounts = account && (gnc_account_n_children(account) != 0);
 +        /* Check here for placeholder accounts, etc. */
 +    }
 +
 +    action_group = gnc_plugin_page_get_action_group(GNC_PLUGIN_PAGE(page));
 +    gnc_plugin_update_actions (action_group, actions_requiring_account_rw,
 +                               "sensitive", is_readwrite && sensitive);
 +    gnc_plugin_update_actions (action_group, actions_requiring_account_always,
 +                               "sensitive", sensitive);
 +    g_signal_emit (page, plugin_page_signals[ACCOUNT_SELECTED], 0, account);
 +
 +    action = gtk_action_group_get_action (action_group, "EditRenumberSubaccountsAction");
 +    g_object_set (G_OBJECT(action), "sensitive",
 +                  is_readwrite && sensitive && subaccounts, NULL);
 +
 +    gnc_plugin_update_actions (action_group, actions_requiring_account_rw,
 +                               "sensitive", is_readwrite && sensitive);
 +    gnc_plugin_update_actions (action_group, actions_requiring_account_always,
 +                               "sensitive", sensitive);
 +}
 +
 +
 +/* Command callbacks */
 +static void
 +gnc_plugin_page_account_tree_cmd_new_account (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account = gnc_plugin_page_account_tree_get_current_account (page);
 +    GtkWindow *parent = GTK_WINDOW (gnc_plugin_page_get_window (GNC_PLUGIN_PAGE (page)));
 +    gnc_ui_new_account_window (parent, gnc_get_current_book(),
 +                               account);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_file_new_hierarchy (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    gnc_ui_hierarchy_assistant(FALSE);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_open_account (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gppat_open_account_common (page, account, FALSE);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_open_subaccounts (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gppat_open_account_common (page, account, TRUE);
 +}
 +
 +
 +/*################## Added for Reg2 #################*/
 +/* Register Firing - Single Account to start with    */
 +static void
 +gnc_plugin_page_account_tree_cmd_open2_account (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gppat_open2_account_common (page, account, FALSE);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_open2_subaccounts (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +
 +    g_return_if_fail (GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE (page));
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gppat_open2_account_common (page, account, TRUE);
 +}
 +/*################## Added for Reg2 #################*/
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_edit_account (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +    GtkWindow *parent = GTK_WINDOW (gnc_plugin_page_get_window (GNC_PLUGIN_PAGE (page)));
 +    ENTER("action %p, page %p", action, page);
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    g_return_if_fail (account != NULL);
 +
 +    gnc_ui_edit_account_window (parent, account);
 +    LEAVE(" ");
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_find_account (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    GtkWidget *window;
 +
 +    ENTER("action %p, page %p", action, page);
 +
 +    window = gnc_plugin_page_get_window(GNC_PLUGIN_PAGE(page));
 +
 +    gnc_find_account_dialog (window, NULL);
 +    LEAVE(" ");
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_find_account_popup (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account = NULL;
 +    GtkWidget *window;
 +
 +    ENTER("action %p, page %p", action, page);
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +
 +    window = gnc_plugin_page_get_window(GNC_PLUGIN_PAGE(page));
 +
 +    gnc_find_account_dialog (window, account);
 +    LEAVE(" ");
 +}
 +
 +/********************************************************************
 + * delete_account_helper
 + * See if this account has any splits present.  Set the user data
 + * and return the same value to stop walking the account tree if
 + * appropriate.
 + ********************************************************************/
 +typedef struct _delete_helper
 +{
 +    gboolean has_splits;
 +    gboolean has_ro_splits;
 +} delete_helper_t;
 +
 +static gpointer
 +delete_account_helper (Account * account, gpointer data)
 +{
 +    delete_helper_t *helper_res = data;
 +    GList *splits;
 +
 +    splits = xaccAccountGetSplitList (account);
 +    if (splits)
 +    {
 +        helper_res->has_splits = TRUE;
 +        while (splits)
 +        {
 +            Split *s = splits->data;
 +            Transaction *txn = xaccSplitGetParent (s);
 +            if (xaccTransGetReadOnly (txn))
 +            {
 +                helper_res->has_ro_splits = TRUE;
 +                break;
 +            }
 +            splits = splits->next;
 +        }
 +    }
 +
 +    return GINT_TO_POINTER (helper_res->has_splits || helper_res->has_ro_splits);
 +}
 +
 +/***
 + *** The OK button of a Delete Account dialog is insensitive if
 + *** and only if a sensitive account selector contains no accounts.
 + ***/
 +static void
 +set_ok_sensitivity(GtkWidget *dialog)
 +{
 +    GtkWidget *button;
 +    GtkWidget *sa_mas, *trans_mas;
 +    gint sa_mas_cnt, trans_mas_cnt;
 +    gboolean sensitive;
 +
 +    sa_mas = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_SA_MAS);
 +    trans_mas = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_TRANS_MAS);
 +    sa_mas_cnt = gnc_account_sel_get_num_account(GNC_ACCOUNT_SEL(sa_mas));
 +    trans_mas_cnt = gnc_account_sel_get_num_account(GNC_ACCOUNT_SEL(trans_mas));
 +
 +    sensitive = (((NULL == sa_mas) ||
 +                  (!gtk_widget_is_sensitive(sa_mas) || sa_mas_cnt)) &&
 +                 ((NULL == trans_mas) ||
 +                  (!gtk_widget_is_sensitive(trans_mas) || trans_mas_cnt)));
 +
 +    button = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_OK_BUTTON);
 +    gtk_widget_set_sensitive(button, sensitive);
 +}
 +
 +static void
 +gppat_populate_gas_list(GtkWidget *dialog,
 +                        GNCAccountSel *gas,
 +                        gboolean exclude_subaccounts)
 +{
 +    Account *account;
 +    GList *filter;
 +
 +    g_return_if_fail(GTK_IS_DIALOG(dialog));
 +    if (gas == NULL)
 +        return;
 +    account = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_ACCOUNT);
 +    filter = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_FILTER);
 +
 +    /* Setting the account type filter triggers GNCAccountSel population. */
 +    gnc_account_sel_set_acct_filters (gas, filter, NULL);
 +
 +    /* Accounts to be deleted must be removed. */
 +    gnc_account_sel_purge_account( gas, account, exclude_subaccounts);
 +
 +    /* The sensitivity of the OK button needs to be reevaluated. */
 +    set_ok_sensitivity(dialog);
 +}
 +
 +void
 +gppat_populate_trans_mas_list(GtkToggleButton *sa_mrb,
 +                              GtkWidget *dialog)
 +{
 +    GtkWidget *trans_mas;
 +
 +    g_return_if_fail(GTK_IS_DIALOG(dialog));
 +
 +    /* Cannot move transactions to subaccounts if they are to be deleted. */
 +    trans_mas = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_TRANS_MAS);
 +    gppat_populate_gas_list(dialog, GNC_ACCOUNT_SEL(trans_mas), !gtk_toggle_button_get_active(sa_mrb));
 +}
 +
 +/* Note that the emitting object (the toggle button) and the signal data
 + * are swapped in below callback function. This is a gtkbuilder feature:
 + * it swaps if you explicitly set an object for a signal handler in the
 + * gtkbuilder xml file.
 + */
 +void
 +gppat_set_insensitive_iff_rb_active(GtkWidget *widget, GtkToggleButton *b)
 +{
 +    GtkWidget *dialog = gtk_widget_get_toplevel(widget);
 +    GtkWidget *subaccount_trans = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_SA_TRANS);
 +    GtkWidget *sa_mas = g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_SA_MAS);
 +    gboolean have_splits = GPOINTER_TO_INT (g_object_get_data(G_OBJECT(dialog), DELETE_DIALOG_SA_SPLITS));
 +
 +    gtk_widget_set_sensitive(widget, !gtk_toggle_button_get_active(b));
 +
 +    // If we have subaccount splits & delete subaccounts, enable subaccount_trans
 +    if ((have_splits) && !gtk_widget_is_sensitive(sa_mas))
 +        gtk_widget_set_sensitive(subaccount_trans, TRUE);
 +    else
 +        gtk_widget_set_sensitive(subaccount_trans, FALSE);
 +
 +    set_ok_sensitivity(dialog);
 +}
 +
 +static GtkWidget *
 +gppat_setup_account_selector (GtkBuilder *builder, GtkWidget *dialog,
 +                              const gchar *hbox, const gchar *sel_name)
 +{
 +    GtkWidget *selector = gnc_account_sel_new();
 +    GtkWidget *box = GTK_WIDGET(gtk_builder_get_object (builder, hbox));
 +
 +    gtk_box_pack_start (GTK_BOX(box), selector, TRUE, TRUE, 0);
 +    g_object_set_data(G_OBJECT(dialog), sel_name, selector);
 +
 +    gppat_populate_gas_list(dialog, GNC_ACCOUNT_SEL(selector), TRUE);
 +    gtk_widget_show_all(box);
 +
 +    return selector;
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_delete_account (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gchar *acct_name;
 +    delete_helper_t delete_res = { FALSE, FALSE };
 +    GtkWidget *window;
 +    GtkWidget *trans_mas = NULL; /* transaction move to account selector */
 +    GtkWidget *sa_mas = NULL;    /* subaccount move to account selector */
 +    GtkWidget *sa_trans_mas = NULL; /* subaccount's transaction move to account selector */
 +    Account *ta = NULL; /* transaction adopter */
 +    Account *saa = NULL; /* subaccount adopter */
 +    Account *sta = NULL; /* subaccount transaction adopter */
 +    GList *splits;
 +    GList* list;
 +    gint response;
 +
 +    if (NULL == account)
 +        return;
 +
 +    /* If the account has objects referring to it, show the list - the account can't be deleted until these
 +       references are dealt with. */
 +    list = qof_instance_get_referring_object_list(QOF_INSTANCE(account));
 +    if (list != NULL)
 +    {
- #define EXPLANATION "The list below shows objects which make use of the account which you want to delete.\nBefore you can delete it, you must either delete those objects or else modify them so they make use\nof another account"
++#define EXPLANATION _("The list below shows objects which make use of the account which you want to delete.\nBefore you can delete it, you must either delete those objects or else modify them so they make use\nof another account")
 +
-         gnc_ui_object_references_show( _(EXPLANATION), list);
++        gnc_ui_object_references_show(EXPLANATION, list);
 +        g_list_free(list);
 +        return;
 +    }
 +
 +    window = gnc_plugin_page_get_window(GNC_PLUGIN_PAGE(page));
 +    acct_name = gnc_account_get_full_name(account);
 +    if (!acct_name)
 +    {
 +        acct_name = g_strdup (_("(no name)"));
 +    }
 +
 +    splits = xaccAccountGetSplitList(account);
 +
 +    /*
 +     * If the account has transactions or child accounts then present a
 +     * dialog to allow the user to specify what should be done with them.
 +     */
 +    if ((NULL != splits) || (gnc_account_n_children(account) > 0))
 +    {
 +        GList *filter = NULL;
 +        GtkBuilder *builder = NULL;
 +        GtkWidget *dialog = NULL;
 +        GtkWidget *widget = NULL;
 +        gchar *title = NULL;
 +
 +        builder = gtk_builder_new();
 +        gnc_builder_add_from_file (builder, "dialog-account.glade", "account_delete_dialog");
 +
 +        dialog = GTK_WIDGET(gtk_builder_get_object (builder, "account_delete_dialog"));
 +        gtk_window_set_transient_for(GTK_WINDOW(dialog), GTK_WINDOW(window));
 +
 +        widget = GTK_WIDGET(gtk_builder_get_object (builder, "header"));
 +        title = g_strdup_printf(_("Deleting account %s"), acct_name);
 +        gtk_label_set_text(GTK_LABEL(widget), title);
 +        g_free(title);
 +
 +        widget = GTK_WIDGET(gtk_builder_get_object (builder, DELETE_DIALOG_OK_BUTTON));
 +        g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_OK_BUTTON, widget);
 +
 +        /*
 +         * Reparent only to accounts of the same
 +         * type as the one being deleted.
 +         */
 +        filter = g_list_prepend(NULL, (gpointer)xaccAccountGetType(account));
 +        g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_FILTER, filter);
 +        g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_ACCOUNT, account);
 +
 +        // Add the account selectors and enable sections as appropiate
 +        // setup transactions selector
 +        trans_mas = gppat_setup_account_selector (builder, dialog, "trans_mas_hbox", DELETE_DIALOG_TRANS_MAS);
 +
 +        // Does the selected account have splits
 +        if (splits)
 +        {
 +            delete_helper_t delete_res2 = { FALSE, FALSE };
 +
 +            delete_account_helper(account, &delete_res2);
 +            if (delete_res2.has_ro_splits)
 +            {
 +                gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "trans_rw")));
 +                widget = GTK_WIDGET(gtk_builder_get_object (builder, "trans_drb"));
 +                gtk_widget_set_sensitive(widget, FALSE);
 +            }
 +            else
 +                gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "trans_ro")));
 +        }
 +        else
 +        {
 +            gtk_widget_set_sensitive (GTK_WIDGET(gtk_builder_get_object (builder, "transactions")), FALSE);
 +            gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "trans_ro")));
 +        }
 +
 +        // setup subaccount account selector
 +        sa_mas = gppat_setup_account_selector (builder, dialog, "sa_mas_hbox", DELETE_DIALOG_SA_MAS);
 +
 +        // setup subaccount transaction selector
 +        sa_trans_mas = gppat_setup_account_selector (builder, dialog, "sa_trans_mas_hbox", DELETE_DIALOG_SA_TRANS_MAS);
 +        g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_SA_TRANS,
 +                          GTK_WIDGET(gtk_builder_get_object (builder, "subaccount_trans")));
 +
 +        // Does the selected account have sub accounts
 +        if (gnc_account_n_children(account) > 0)
 +        {
 +            // Check for RO txns in descendants
 +            gnc_account_foreach_descendant_until(account, delete_account_helper,
 +                                                 &delete_res);
 +            if (delete_res.has_splits)
 +            {
 +                if (delete_res.has_ro_splits)
 +                {
 +                    gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "sa_trans_rw")));
 +                    widget = GTK_WIDGET(gtk_builder_get_object (builder, "sa_trans_drb"));
 +                    gtk_widget_set_sensitive(widget, FALSE);
 +                }
 +                else
 +                    gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "sa_trans_ro")));
 +
 +                g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_SA_SPLITS, GINT_TO_POINTER(1));
 +            }
 +            else
 +            {
 +                g_object_set_data(G_OBJECT(dialog), DELETE_DIALOG_SA_SPLITS, GINT_TO_POINTER(0));
 +                gtk_widget_set_sensitive (GTK_WIDGET(gtk_builder_get_object (builder, "subaccount_trans")), FALSE);
 +                gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "sa_trans_ro")));
 +            }
 +        }
 +        else
 +        {
 +            gtk_widget_set_sensitive(GTK_WIDGET(gtk_builder_get_object (builder, "subaccounts")), FALSE);
 +            gtk_widget_set_sensitive(GTK_WIDGET(gtk_builder_get_object (builder, "subaccount_trans")), FALSE);
 +            gtk_widget_hide(GTK_WIDGET(gtk_builder_get_object (builder, "sa_trans_ro")));
 +        }
 +
 +        /* default to cancel */
 +        gtk_dialog_set_default_response (GTK_DIALOG(dialog), GTK_RESPONSE_CANCEL);
 +
 +        gtk_builder_connect_signals(builder, dialog);
 +        g_object_unref(G_OBJECT(builder));
 +
 +        /*
 +         * Note that one effect of the modal dialog is preventing
 +         * the account selectors from being repopulated.
 +         */
 +        response = gtk_dialog_run(GTK_DIALOG(dialog));
 +        if (GTK_RESPONSE_ACCEPT != response)
 +        {
 +            /* Account deletion is cancelled, so clean up and return. */
 +            gtk_widget_destroy(dialog);
 +            g_list_free(filter);
 +            g_free(acct_name);
 +            return;
 +        }
 +        if (trans_mas && gtk_widget_is_sensitive(trans_mas))
 +            ta = gnc_account_sel_get_account(GNC_ACCOUNT_SEL(trans_mas));
 +        if (sa_mas && gtk_widget_is_sensitive(sa_mas))
 +            saa = gnc_account_sel_get_account(GNC_ACCOUNT_SEL(sa_mas));
 +        if (sa_trans_mas && gtk_widget_is_sensitive(sa_trans_mas))
 +            sta = gnc_account_sel_get_account(GNC_ACCOUNT_SEL(sa_trans_mas));
 +        gtk_widget_destroy(dialog);
 +        g_list_free(filter);
 +    } /* (NULL != splits) || (NULL != children) */
 +
 +    /*
 +     * Present a message to the user which specifies what will be
 +     * deleted and what will be reparented, then ask for verification.
 +     */
 +    {
 +        const char *format = _("The account %s will be deleted.");
 +        char *lines[8];
 +        char *message;
 +        char *name;
 +        int i = 0;
 +        GtkWidget *dialog;
 +
 +        lines[0] = g_strdup_printf(format, acct_name);
 +        if (splits)
 +        {
 +            if (ta)
 +            {
 +                name = gnc_account_get_full_name(ta);
 +                format = _("All transactions in this account will be moved to "
 +                           "the account %s.");
 +                lines[++i] = g_strdup_printf(format, name);
 +            }
 +            else if (splits)
 +            {
 +                format = _("All transactions in this account will be deleted.");
 +                lines[++i] = g_strdup_printf("%s", format);
 +            }
 +        }
 +        if (gnc_account_n_children(account) > 0)
 +        {
 +            if (saa)
 +            {
 +                name = gnc_account_get_full_name(saa);
 +                format = _("All of its sub-accounts will be moved to "
 +                           "the account %s.");
 +                lines[++i] = g_strdup_printf(format, name);
 +            }
 +            else
 +            {
 +                format = _("All of its subaccounts will be deleted.");
 +                lines[++i] = g_strdup_printf("%s", format);
 +                if (sta)
 +                {
 +                    name = gnc_account_get_full_name(sta);
 +                    format = _("All sub-account transactions will be moved to "
 +                               "the account %s.");
 +                    lines[++i] = g_strdup_printf(format, name);
 +                }
 +                else if (delete_res.has_splits)
 +                {
 +                    format = _("All sub-account transactions will be deleted.");
 +                    lines[++i] = g_strdup_printf("%s", format);
 +                }
 +            }
 +        }
 +        lines[++i] = _("Are you sure you want to do this?");
 +        lines[i] = NULL;
 +        i--; /* Don't try to free the constant question. */
 +        message = g_strjoinv(" ", lines);
 +        while (i--)
 +        {
 +            g_free(lines[i]);
 +        }
 +
 +        dialog =  gtk_message_dialog_new(GTK_WINDOW(window),
 +                                         GTK_DIALOG_DESTROY_WITH_PARENT,
 +                                         GTK_MESSAGE_QUESTION,
 +                                         GTK_BUTTONS_NONE,
 +                                         "%s", message);
 +        g_free(message);
 +        gtk_dialog_add_buttons(GTK_DIALOG(dialog),
 +                               _("_Cancel"), GTK_RESPONSE_CANCEL,
 +                               _("_Delete"), GTK_RESPONSE_ACCEPT,
 +                               (gchar *)NULL);
 +        gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_CANCEL);
 +        response = gtk_dialog_run(GTK_DIALOG(dialog));
 +        gtk_widget_destroy(dialog);
 +
 +        if (GTK_RESPONSE_ACCEPT == response)
 +        {
 +            GList *acct_list, *ptr;
 +            const GncGUID *guid;
 +            gchar guidstr[GUID_ENCODING_LENGTH+1];
 +
 +            gnc_set_busy_cursor(NULL, TRUE);
 +            gnc_suspend_gui_refresh ();
 +
 +            /* Move subaccounts and transactions if this was requested */
 +            xaccAccountBeginEdit (account);
 +            if (NULL != saa)
 +            {
 +
 +                xaccAccountBeginEdit (saa);
 +                acct_list = gnc_account_get_children(account);
 +                for (ptr = acct_list; ptr; ptr = g_list_next(ptr))
 +                    gnc_account_append_child (saa, ptr->data);
 +                g_list_free(acct_list);
 +                xaccAccountCommitEdit (saa);
 +            }
 +            else if (NULL != sta)
 +            {
 +                /* Move the splits of its subaccounts, if any. */
 +                gnc_account_foreach_descendant(account,
 +                                               (AccountCb)xaccAccountMoveAllSplits,
 +                                               sta);
 +            }
 +            if (NULL != ta)
 +            {
 +                /* Move the splits of the account to be deleted. */
 +                xaccAccountMoveAllSplits (account, ta);
 +            }
 +            xaccAccountCommitEdit (account);
 +
 +            /* Drop all references from the state file for
 +             * any subaccount the account still has
 +             */
 +            acct_list = gnc_account_get_children(account);
 +            for (ptr = acct_list; ptr; ptr = g_list_next(ptr))
 +            {
 +                guid = xaccAccountGetGUID (ptr->data);
 +                guid_to_string_buff (guid, guidstr);
 +                gnc_state_drop_sections_for (guidstr);
 +            }
 +            g_list_free(acct_list);
 +
 +            /* Drop all references from the state file for this account
 +             */
 +            guid = xaccAccountGetGUID (account);
 +            guid_to_string_buff (guid, guidstr);
 +            gnc_state_drop_sections_for (guidstr);
 +
 +            /*
 +             * Finally, delete the account, any subaccounts it may still
 +             * have, and any splits it or its subaccounts may still have.
 +             */
 +            xaccAccountBeginEdit (account);
 +            xaccAccountDestroy (account);
 +            gnc_resume_gui_refresh ();
 +            gnc_unset_busy_cursor(NULL);
 +        }
 +    }
 +    g_free(acct_name);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_renumber_accounts (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +    GtkWidget *window;
 +
 +    window = gnc_plugin_page_get_window(GNC_PLUGIN_PAGE(page));
 +    account = gnc_plugin_page_account_tree_get_current_account(page);
 +    if (!window || !account)
 +        return;
 +
 +    gnc_account_renumber_create_dialog(window, account);
 +}
 +
 +/*********************/
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_view_filter_by (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    GncPluginPageAccountTreePrivate *priv;
 +
 +    g_return_if_fail(GNC_IS_PLUGIN_PAGE_ACCOUNT_TREE(page));
 +    ENTER("(action %p, page %p)", action, page);
 +
 +    priv = GNC_PLUGIN_PAGE_ACCOUNT_TREE_GET_PRIVATE(page);
 +    account_filter_dialog_create(&priv->fd, GNC_PLUGIN_PAGE(page));
 +    LEAVE(" ");
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_reconcile (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkWidget *window;
 +    Account *account;
 +    RecnWindow *recnData;
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    g_return_if_fail (account != NULL);
 +
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    recnData = recnWindow (window, account);
 +    gnc_ui_reconcile_window_raise (recnData);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_autoclear (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkWidget *window;
 +    Account *account;
 +    AutoClearWindow *autoClearData;
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    g_return_if_fail (account != NULL);
 +
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    autoClearData = autoClearWindow (window, account);
 +    gnc_ui_autoclear_window_raise (autoClearData);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_transfer (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkWidget *window;
 +    Account *account;
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    gnc_xfer_dialog (window, account);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_stock_split (GtkAction *action,
 +        GncPluginPageAccountTree *page)
 +{
 +    GtkWidget *window;
 +    Account *account;
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    window = GNC_PLUGIN_PAGE (page)->window;
 +    gnc_stock_split_dialog (window, account);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_lots (GtkAction *action,
 +                                       GncPluginPageAccountTree *page)
 +{
 +    Account *account;
 +
 +    account = gnc_plugin_page_account_tree_get_current_account (page);
 +    gnc_lot_viewer_dialog (account);
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_scrub (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account = gnc_plugin_page_account_tree_get_current_account (page);
 +    GncWindow *window;
 +
 +    g_return_if_fail (account != NULL);
 +
 +    gnc_suspend_gui_refresh ();
 +
 +    window = GNC_WINDOW(GNC_PLUGIN_PAGE (page)->window);
 +    gnc_window_set_progressbar_window (window);
 +
 +    xaccAccountScrubOrphans (account, gnc_window_show_progress);
 +    xaccAccountScrubImbalance (account, gnc_window_show_progress);
 +
 +    // XXX: Lots/capital gains scrubbing is disabled
 +    if (g_getenv("GNC_AUTO_SCRUB_LOTS") != NULL)
 +        xaccAccountScrubLots(account);
 +
 +    gncScrubBusinessAccount(account, gnc_window_show_progress);
 +
 +
 +    gnc_resume_gui_refresh ();
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_scrub_sub (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *account = gnc_plugin_page_account_tree_get_current_account (page);
 +    GncWindow *window;
 +
 +    g_return_if_fail (account != NULL);
 +
 +    gnc_suspend_gui_refresh ();
 +
 +    window = GNC_WINDOW(GNC_PLUGIN_PAGE (page)->window);
 +    gnc_window_set_progressbar_window (window);
 +
 +    xaccAccountTreeScrubOrphans (account, gnc_window_show_progress);
 +    xaccAccountTreeScrubImbalance (account, gnc_window_show_progress);
 +
 +    // XXX: Lots/capital gains scrubbing is disabled
 +    if (g_getenv("GNC_AUTO_SCRUB_LOTS") != NULL)
 +        xaccAccountTreeScrubLots(account);
 +
 +    gncScrubBusinessAccountTree(account, gnc_window_show_progress);
 +
 +    gnc_resume_gui_refresh ();
 +}
 +
 +static void
 +gnc_plugin_page_account_tree_cmd_scrub_all (GtkAction *action, GncPluginPageAccountTree *page)
 +{
 +    Account *root = gnc_get_current_root_account ();
 +    GncWindow *window;
 +
 +    gnc_suspend_gui_refresh ();
 +
 +    window = GNC_WINDOW(GNC_PLUGIN_PAGE (page)->window);
 +    gnc_window_set_progressbar_window (window);
 +
 +    xaccAccountTreeScrubOrphans (root, gnc_window_show_progress);
 +    xaccAccountTreeScrubImbalance (root, gnc_window_show_progress);
 +    // XXX: Lots/capital gains scrubbing is disabled
 +    if (g_getenv("GNC_AUTO_SCRUB_LOTS") != NULL)
 +        xaccAccountTreeScrubLots(root);
 +
 +    gncScrubBusinessAccountTree(root, gnc_window_show_progress);
 +
 +    gnc_resume_gui_refresh ();
 +}
 +
 +/** @} */
 +/** @} */
diff --cc libgnucash/app-utils/date-utilities.scm
index 1729d08,0000000..1e6b2f6
mode 100644,000000..100644
--- a/libgnucash/app-utils/date-utilities.scm
+++ b/libgnucash/app-utils/date-utilities.scm
@@@ -1,1072 -1,0 +1,1072 @@@
 +;; date-utilities.scm -- date utility functions.
 +;; Bryan Larsen (blarsen at ada-works.com)
 +;; Revised by Christopher Browne
 +;; Improvement to financial year support by Yves-Eric Martin
 +;;
 +;; 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 at gnu.org
 +
 +
 +(use-modules (gnucash core-utils)
 +             (gnucash gettext))
 +
 +(define gnc:reldate-list '())
 +
 +;; get stuff from localtime date vector
 +(define (gnc:date-get-year datevec)
 +  (+ 1900 (tm:year datevec)))
 +(define (gnc:date-get-quarter datevec)
 +  (+ (quotient (tm:mon datevec) 3) 1))
 +(define (gnc:date-get-month-day datevec)
 +  (tm:mday datevec))
 +;; get month with january==1
 +(define (gnc:date-get-month datevec)
 +  (+ (tm:mon datevec) 1))
 +(define (gnc:date-get-week-day datevec)
 +  (+ (tm:wday datevec) 1))
 +;; jan 1 == 1
 +(define (gnc:date-get-week datevec)
 +  (gnc:date-to-week (gnc:time64-start-day-time
 +                     (gnc-mktime datevec))))
 +
 +(define (gnc:date-get-year-day datevec)
 +  (+ (tm:yday datevec) 1))
 +
 +(define (gnc:time64-get-year t64)
 +  (gnc:date-get-year (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-quarter t64)
 +  (gnc:date-get-quarter (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-month-day t64)
 +  (gnc:date-get-month-day (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-month t64)
 +  (gnc:date-get-month (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-week-day t64)
 +  (gnc:date-get-week-day (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-week t64)
 +  (gnc:date-get-week (gnc-localtime t64)))
 +
 +(define (gnc:time64-get-year-day t64)
 +  (gnc:date-get-year-day (gnc-localtime t64)))
 +
 +(define (gnc:date-get-year-string datevec)
 +  (gnc-locale-to-utf8 (strftime "%Y" datevec)))
 +
 +(define (gnc:date-get-quarter-string datevec)
 +  (format #f "Q~d" (gnc:date-get-quarter datevec)))
 +
 +(define (gnc:date-get-quarter-year-string datevec)
 +  (string-append 
 +   (gnc:date-get-quarter-string datevec) 
 +   " " 
 +   (gnc:date-get-year-string datevec)))
 +
 +(define (gnc:date-get-month-string datevec)
 +  (gnc-locale-to-utf8 (strftime "%B" datevec)))
 +
 +(define (gnc:date-get-month-year-string datevec)
 +  (gnc-locale-to-utf8 (strftime "%B %Y" datevec)))
 +
 +(define (gnc:date-get-week-year-string datevec)
 +  (let* ((beginweekt64 (* (gnc:time64-get-week
 +                            (gnc-mktime datevec))
 +                          604800))
 +         (begin-string (qof-print-date (+ beginweekt64 345600)))
 +         (end-string (qof-print-date (+ beginweekt64 864000))))
 +    (format #f (_ "~a to ~a") begin-string end-string)))
 +
 +;; is leap year?
 +(define (gnc:leap-year? year)
 +  (or (and (zero? (remainder year 4))
 +           (not (zero? (remainder year 100))))
 +      (zero? (remainder year 400))))
 +
 +;; number of days in year
 +(define (gnc:days-in-year year)
 +  (if (gnc:leap-year? year) 366 365))
 +
 +;; number of days in month
 +(define (gnc:days-in-month month year)
 +  (case month
 +    ((1 3 5 7 8 10 12) 31)
 +    ((4 6 9 11) 30)
 +    ((2) (if (gnc:leap-year? year) 29 28))))
 +
 +;; convert a date in seconds since 1970 into # of years since 1970 as
 +;; a fraction.
 +(define (gnc:date-to-year-fraction caltime)
 +  (let ((lt (gnc-localtime caltime)))
 +    (+ (- (gnc:date-get-year lt) 1970)
 +       (/ (- (gnc:date-get-year-day lt) 1.0)
 +	  (* 1.0 (gnc:days-in-year (gnc:date-get-year lt)))))))
 +
 +;; return the number of years (in floating point format) between two dates.
 +(define (gnc:date-year-delta caltime1 caltime2)
 +  (let* ((lt1 (gnc-localtime caltime1))
 +	 (lt2 (gnc-localtime caltime2))
 +	 (day1 (gnc:date-get-year-day lt1))
 +	 (day2 (gnc:date-get-year-day lt2))
 +	 (year1 (gnc:date-get-year lt1))
 +	 (year2 (gnc:date-get-year lt2))
 +	 (dayadj1 (if (and (not (gnc:leap-year? year1))
 +			   (>= day1 59))
 +		      (+ day1 1)
 +		      day1))
 +	 (dayadj2 (if (and (not (gnc:leap-year? year2))
 +			   (>= day2 59))
 +		      (+ day2 1)
 +		      day2)))
 +    (+ (- (gnc:date-get-year lt2) (gnc:date-get-year lt1))
 +       (/ (- dayadj2 dayadj1) 
 +	  366.0))))
 +
 +;; convert a date in seconds since 1970 into # of 1/2 years since 1970 as
 +;; a fraction (doubling date-to-year-fraction)
 +(define (gnc:date-to-halfyear-fraction caltime)
 +  (* (gnc:date-to-year-fraction caltime) 2))
 +
 +;; convert a date in seconds since 1970 into # of quarters since 1970
 +;; (assuming quarter = 3 months and using 1/3 of date-to-month-fraction)
 +(define (gnc:date-to-quarter-fraction caltime)
 +  (/ (gnc:date-to-month-fraction caltime) 3))
 +
 +;; convert a date in seconds since 1970 into # of months since 1970
 +(define (gnc:date-to-month-fraction caltime)
 +  (let ((lt (gnc-localtime caltime)))
 +    (+ (* 12 (- (gnc:date-get-year lt) 1970.0))
 +       (gnc:date-get-month lt) -1
 +       (/ (- (gnc:date-get-month-day lt) 1.0) (gnc:days-in-month 
 +					       (gnc:date-get-month lt)
 +					       (gnc:date-get-year lt))))))
 +
 +;; convert a date in seconds since 1970 into # of two-week periods since
 +;; Jan 4, 1970 ignoring leap-seconds (just halfing date-to-week-fraction)
 +(define (gnc:date-to-twoweek-fraction caltime)
 +  (/ (gnc:date-to-week-fraction caltime) 2))
 +
 +;; convert a date in seconds since 1970 into # of weeks since Jan 4, 1970
 +;; ignoring leap-seconds
 +(define (gnc:date-to-week-fraction caltime)
 +  (/ (- (/ (/ caltime 3600.0) 24) 3) 7))
 +
 +(define (gnc:date-to-week caltime)
 +  (floor (/ (- (/ caltime 86400) 3) 7)))
 +
 +;; convert a date in seconds since 1970 into # of days since Feb 28, 1970
 +;; ignoring leap-seconds
 +(define (gnc:date-to-day-fraction caltime)
 +  (- (/ (/ caltime 3600.0) 24) 59))
 +
 +;; Returns the function that converts a date into a fraction of
 +;; {year,month,week,day} according to the given symbol, or #f if the
 +;; symbol was unknown
 +(define (gnc:date-get-fraction-func interval)
 +  (case interval
 +    ((YearDelta) gnc:date-to-year-fraction)
 +    ((HalfYearDelta) gnc:date-to-halfyear-fraction)
 +    ((QuarterDelta) gnc:date-to-quarter-fraction)
 +    ((MonthDelta) gnc:date-to-month-fraction)
 +    ((TwoWeekDelta) gnc:date-to-twoweek-fraction)
 +    ((WeekDelta) gnc:date-to-week-fraction)
 +    ((DayDelta) gnc:date-to-day-fraction)
 +    (else #f)))
 +
 +;; Modify a date
 +(define (moddate op adate delta)
 +  (let ((newtm (gnc-localtime adate)))
 +    (begin
 +      (set-tm:sec newtm (op (tm:sec newtm) (tm:sec delta)))
 +      (set-tm:min newtm (op (tm:min newtm) (tm:min delta)))
 +      (set-tm:hour newtm (op (tm:hour newtm) (tm:hour delta)))
 +      (set-tm:mday newtm (op (tm:mday newtm) (tm:mday delta)))
 +      (set-tm:mon newtm (op (tm:mon newtm) (tm:mon delta)))
 +      (set-tm:year newtm (op (tm:year newtm) (tm:year delta)))
 +      (set-tm:isdst newtm -1)
 +      (gnc-mktime newtm))))
 +
 +;; Add or subtract time from a date
 +(define (decdate adate delta) (moddate - adate delta ))
 +(define (incdate adate delta) (moddate + adate delta ))
 +
 +;; date-granularity comparison functions.
 +
 +(define (gnc:time64-le-date t1 t2)
 +  (<= (time64CanonicalDayTime t1)
 +      (time64CanonicalDayTime t2)))
 +
 +(define (gnc:time64-ge-date t1 t2)
 +  (gnc:time64-le-date t2 t1))
 +
 +;; Build a list of time intervals. 
 +;;
 +;; Note that the last interval will be shorter than <incr> if
 +;; (<curd>-<endd>) is not an integer multiple of <incr>. If you don't
 +;; want that you'll have to write another function.
 +(define (gnc:make-date-interval-list current-date end-date increment)
 +  (if (< current-date end-date)
 +      (let ((next-date (incdate current-date increment)))
 +        (if (< next-date end-date)
 +            (cons (list current-date (decdate next-date SecDelta) '())
 +                  (gnc:make-date-interval-list next-date end-date increment))
 +            (cons (list current-date end-date '())
 +                  '())))
 +      '()))
 +  
 +;; Build a list of times.  The dates are evenly spaced with the
 +;; stepsize 'incr'. If the difference of 'startdate' and 'enddate' is
 +;; not an integer multiple of 'incr', 'enddate' will be added as the
 +;; last element of the list, thus making the last interval smaller
 +;; than 'incr'.
 +(define (gnc:make-date-list startdate enddate incr)
 +  (if (< startdate enddate)
 +      (cons startdate
 +            (gnc:make-date-list (incdate startdate incr)
 +                                enddate incr))
 +      (list enddate)))
 +
 +; A reference zero date - the Beginning Of The Epoch
 +; Note: use of eval is evil... by making this a generator function, 
 +; each delta function gets its own instance of Zero Date
 +(define (make-zdate) 
 +  (let ((zd (gnc-localtime (current-time))))
 +    (set-tm:hour zd 0)
 +    (set-tm:min zd 0)
 +    (set-tm:sec zd 0)
 +    (set-tm:mday zd 0)
 +    (set-tm:mon zd 0)
 +    (set-tm:year zd 0)
 +    (set-tm:yday zd 0)
 +    (set-tm:wday zd 0)
 +    (set-tm:isdst zd 0)
 +    zd))
 +
 +
 +(define SecDelta 
 +  (let ((ddt (make-zdate)))
 +    (set-tm:sec ddt 1)
 +    ddt))
 +
 +(define DayDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mday ddt 1)
 +    ddt))
 +
 +(define WeekDelta 
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mday ddt 7)
 +    ddt))
 +
 +(define TwoWeekDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mday ddt 14)
 +    ddt))
 +
 +(define MonthDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mon ddt 1)
 +    ddt))
 +
 +(define QuarterDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mon ddt 3)
 +    ddt))
 +
 +(define HalfYearDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mon ddt 6)
 +    ddt))
 +
 +(define YearDelta 
 +  (let ((ddt (make-zdate)))
 +    (set-tm:year ddt 1)
 +    ddt))
 +
 +
 +(define ThirtyDayDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mday ddt 30)
 +    ddt))
 +
 +(define NinetyDayDelta
 +  (let ((ddt (make-zdate)))
 +    (set-tm:mday ddt 90)
 +    ddt))
 +
 +;; if you add any more FooDeltas, add to this list!!!
 +
 +(define deltalist
 +  (list (cons 'SecDelta SecDelta)
 +	(cons 'DayDelta DayDelta)
 +	(cons 'WeekDelta WeekDelta)
 +	(cons 'TwoWeekDelta TwoWeekDelta)
 +	(cons 'MonthDelta MonthDelta)
 +	(cons 'QuarterDelta QuarterDelta)
 +	(cons 'HalfYearDelta HalfYearDelta)
 +	(cons 'YearDelta YearDelta)
 +	(cons 'ThirtyDayDelta ThirtyDayDelta)
 +	(cons 'NinetyDayDelta NinetyDayDelta)))
 +
 +(define (gnc:deltasym-to-delta ds)
 +  (let ((retval (assq ds deltalist)))
 +    (if (pair? retval)
 +	(cdr retval)
 +	#f)))
 +
 +;; given a time64 time on a certain day (local time)
 +;; converts it to be midday that day.
 +
 +(define (gnc:time64-start-day-time t64)
 +  (let ((bdt (gnc-localtime t64)))
 +    (set-tm:sec bdt 0)
 +    (set-tm:min bdt 0)
 +    (set-tm:hour bdt 0)
 +    (set-tm:isdst bdt -1)
 +    (gnc-mktime bdt)))
 +
 +(define (gnc:time64-end-day-time t64)
 +  (let ((bdt (gnc-localtime t64)))
 +    (set-tm:sec bdt 59)
 +    (set-tm:min bdt 59)
 +    (set-tm:hour bdt 23)
 +    (set-tm:isdst bdt -1)
 +    (gnc-mktime bdt)))
 +
 +(define (gnc:time64-previous-day t64)
 +  (decdate t64 DayDelta))
 +
 +(define (gnc:time64-next-day t64)
 +  (incdate t64 DayDelta))
 +
 +
 +(define (gnc:reldate-get-symbol x) (vector-ref x 0))
 +(define (gnc:reldate-get-string x) (vector-ref x 1))
 +(define (gnc:reldate-get-desc x) (vector-ref x 2))
 +(define (gnc:reldate-get-fn x) (vector-ref x 3))
 +
 +(define (gnc:make-reldate-hash hash reldate-list)
 +  (map (lambda (reldate) (hash-set! 
 +			  hash 
 +			  (gnc:reldate-get-symbol reldate)
 +			  reldate))
 +       reldate-list))
 +
 +(define gnc:reldate-string-db (gnc:make-string-database))
 +
 +(define gnc:relative-date-values '())
 +
 +(define gnc:relative-date-hash (make-hash-table 23))
 +
 +(define (gnc:get-absolute-from-relative-date date-symbol)
 +  (let ((rel-date-data (hash-ref gnc:relative-date-hash date-symbol)))
 +    (if rel-date-data
 +       ((gnc:reldate-get-fn rel-date-data))
 +       (gnc:error "Tried to look up an undefined date symbol"))))
 +
 +(define (gnc:get-relative-date-strings date-symbol)
 +  (let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
 +    
 +    (cons (gnc:reldate-get-string rel-date-info)
 +	  (gnc:relate-get-desc rel-date-info))))
 +
 +(define (gnc:get-relative-date-string date-symbol)
 +  (let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
 +    (gnc:reldate-get-string rel-date-info)))
 +
 +(define (gnc:get-relative-date-desc date-symbol)
 +  (let ((rel-date-info (hash-ref gnc:relative-date-hash date-symbol)))
 +    (gnc:reldate-get-desc rel-date-info)))
 +
 +(define (gnc:get-start-cal-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:mon now 0)
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-cal-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (set-tm:mday now 31)
 +    (set-tm:mon now 11)
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-start-prev-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:mon now 0)
 +    (set-tm:year now (- (tm:year now) 1))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-prev-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (set-tm:mday now 31)
 +    (set-tm:mon now 11)
 +    (set-tm:year now (- (tm:year now) 1))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-start-next-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:mon now 0)
 +    (set-tm:year now (+ (tm:year now) 1))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-next-year)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (set-tm:mday now 31)
 +    (set-tm:mon now 11)
 +    (set-tm:year now (+ (tm:year now) 1))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-start-accounting-period)
 +  (gnc-accounting-period-fiscal-start))
 +
 +(define (gnc:get-end-accounting-period)
 +  (gnc-accounting-period-fiscal-end))
 +
 +(define (gnc:get-start-this-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-this-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59) 
 +    (set-tm:hour now 23)
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1) 
 +					(+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +    
 +(define (gnc:get-start-prev-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (if (= (tm:mon now) 0)
 +	(begin 
 +	  (set-tm:mon now 11)
 +	  (set-tm:year now (- (tm:year now) 1)))
 +	(set-tm:mon now (- (tm:mon now) 1)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-prev-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59) 
 +    (set-tm:hour now 23)
 +    (if (= (tm:mon now) 0)
 +	(begin
 +	  (set-tm:mon now 11)
 +	  (set-tm:year now (- (tm:year now) 1)))
 +	(set-tm:mon now (- (tm:mon now) 1)))
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1) 
 +					(+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +    
 +(define (gnc:get-start-next-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (if (= (tm:mon now) 11)
 +	(begin 
 +	  (set-tm:mon now 0)
 +	  (set-tm:year now (+ (tm:year now) 1)))
 +	(set-tm:mon now (+ (tm:mon now) 1)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-next-month)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59) 
 +    (set-tm:hour now 23)
 +    (if (= (tm:mon now) 11)
 +	(begin
 +	  (set-tm:mon now 0)
 +	  (set-tm:year now (+ (tm:year now) 1)))
 +	(set-tm:mon now (+ (tm:mon now) 1)))
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1) 
 +					(+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +    
 +(define (gnc:get-start-current-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:mon now (- (tm:mon now) (modulo (tm:mon now) 3)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-current-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (set-tm:mon now (+ (tm:mon now) 
 +		       (- 2 (modulo (tm:mon now) 3))))
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
 +                                        (+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-start-prev-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (set-tm:mon now (- (tm:mon now) (modulo (tm:mon now) 3)))
 +    (if (= (tm:mon now) 0)
 +	(begin
 +	  (set-tm:mon now 9)
 +	  (set-tm:year now (- (tm:year now) 1)))
 +	(set-tm:mon now (- (tm:mon now) 3)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-prev-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (if (< (tm:mon now) 3)
 +	(begin
 +	  (set-tm:mon now 11)
 +	  (set-tm:year now (- (tm:year now) 1)))
 +	(set-tm:mon now (- (tm:mon now) 
 +			     (+ 1 (modulo (tm:mon now) 3)))))
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
 +                                        (+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-start-next-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 0)
 +    (set-tm:min now 0)
 +    (set-tm:hour now 0)
 +    (set-tm:mday now 1)
 +    (if (> (tm:mon now) 8)
 +	(begin
 +	  (set-tm:mon now 0)
 +	  (set-tm:year now (+ (tm:year now) 1)))
 +        (set-tm:mon now (+ (tm:mon now) (- 3 (modulo (tm:mon now) 3)))))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-end-next-quarter)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set-tm:sec now 59)
 +    (set-tm:min now 59)
 +    (set-tm:hour now 23)
 +    (if (> (tm:mon now) 8)
 +	(begin
 +	  (set-tm:mon now 2)
 +	  (set-tm:year now (+ (tm:year now) 1)))
 +	(set-tm:mon now (+ (tm:mon now) 
 +			     (+ 1 (modulo (tm:mon now) 3)))))
 +    (set-tm:mday now (gnc:days-in-month (+ (tm:mon now) 1)
 +                                        (+ (tm:year now) 1900)))
 +    (set-tm:isdst now -1)
 +    (gnc-mktime now)))
 +
 +(define (gnc:get-today)
 +  (current-time))
 +
 +(define (gnc:get-one-month-ago)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (= (tm:mon now) 0)
 +	(begin
 +	  (set-tm:mon now 11)
 +	  (set-tm:year now (- (tm:year now) 1)))
 +	(set-tm:mon now (- (tm:mon now) 1)))
-     (let ((month-length (gnc:days-in-month (+ (tm:mon now) 1)
-                                            (+ (tm:year now) 1900))))
-       (if (> month-length (tm:mday now))
- 	  (set-tm:mday now month-length))
++    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
++                                         (+ (tm:year now) 1900))))
++      (if (< month-days (tm:mday now))
++	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-three-months-ago)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (< (tm:mon now) 3)
 +	(begin
 +	  (set:tm-mon now (+ (tm:mon now) 12))
 +	  (set:tm-year now  (- (tm:year now) 1))))
 +    (set:tm-mon now (- (tm:mon now) 3))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
 +                                         (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-six-months-ago)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (< (tm:mon now) 6)
 +	(begin
 +	  (set:tm-mon now (+ (tm:mon now) 12))
 +	  (set:tm-year now  (- (tm:year now) 1))))
 +    (set:tm-mon now (- (tm:mon now) 6))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
 +                                         (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-one-year-ago)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set:tm-year now (- (tm:year now) 1))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
-                                           (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++                                         (+ (tm:year now) 1900))))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-one-month-ahead)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (= (tm:mon now) 11)
 +	(begin
 +	  (set-tm:mon now 0)
 +	  (set-tm:year now (+ (tm:year now) 1)))
 +	(set-tm:mon now (+ (tm:mon now) 1)))
-     (let ((month-length (gnc:days-in-month (+ (tm:mon now) 1)
-                                            (+ (tm:year now) 1900))))
-       (if (> month-length (tm:mday now))
- 	  (set-tm:mday now month-length))
++    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
++                                         (+ (tm:year now) 1900))))
++      (if (< month-days (tm:mday now))
++	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-three-months-ahead)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (> (tm:mon now) 8)
 +	(begin
 +	  (set:tm-mon now (- (tm:mon now) 9))
 +	  (set:tm-year now  (+ (tm:year now) 1))
-        (set:tm-mon now (+ (tm:mon now) 3))))
++          (set:tm-mon now (+ (tm:mon now) 3))))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
 +                                         (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-six-months-ahead)
 +  (let ((now (gnc-localtime (current-time))))
 +    (if (> (tm:mon now) 5)
 +	(begin
 +	  (set:tm-mon now (- (tm:mon now) 6))
 +	  (set:tm-year now  (+ (tm:year now) 1))
-        (set:tm-mon now (+ (tm:mon now) 6))))
++          (set:tm-mon now (+ (tm:mon now) 6))))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
 +                                         (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +(define (gnc:get-one-year-ahead)
 +  (let ((now (gnc-localtime (current-time))))
 +    (set:tm-year now (+ (tm:year now) 1))
 +    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
-                                           (+ (tm:year now) 1900))))
-       (if (> month-days (tm:mday now))
++                                         (+ (tm:year now) 1900))))
++      (if (< month-days (tm:mday now))
 +	  (set-tm:mday now month-days))
 +      (set-tm:isdst now -1)
 +      (gnc-mktime now))))
 +
 +;; There is no GNC:RELATIVE-DATES list like the one mentioned in
 +;; gnucash-design.info, is there? Here are the currently defined
 +;; items, loosely grouped.
 +;;today
 +;;start-cal-year end-cal-year start-prev-year end-prev-year
 +;;start-this-month end-this-month start-prev-month end-prev-month
 +;;start-current-quarter end-current-quarter start-prev-quarter
 +;;end-prev-quarter
 +;;one-month-ago three-months-ago six-months-ago one-year-ago
 +;;start-cur-fin-year start-prev-fin-year end-prev-fin-year
 +
 +(define (gnc:reldate-initialize)
 +  (gnc:reldate-string-db 
 +   'store 'start-cal-year-string 
 +   (N_ "Start of this year"))
 +  (gnc:reldate-string-db 
 +   'store 'start-cal-year-desc 
 +   (N_ "First day of the current calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-cal-year-string 
 +   (N_ "End of this year"))
 +  (gnc:reldate-string-db 
 +   'store 'end-cal-year-desc 
 +   (N_ "Last day of the current calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-year-string 
 +   (N_ "Start of previous year"))
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-year-desc 
 +   (N_ "First day of the previous calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-year-string 
 +   (N_ "End of previous year"))
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-year-desc 
 +   (N_ "Last day of the previous calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-next-year-string 
 +   (N_ "Start of next year"))
 +  (gnc:reldate-string-db 
 +   'store 'start-next-year-desc 
 +   (N_ "First day of the next calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-next-year-string 
 +   (N_ "End of next year"))
 +  (gnc:reldate-string-db 
 +   'store 'end-next-year-desc 
 +   (N_ "Last day of the next calendar year."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-accounting-period-string 
 +   (N_ "Start of accounting period"))
 +  (gnc:reldate-string-db 
 +   'store 'start-accounting-period-desc 
 +   (N_ "First day of the accounting period, as set in the global preferences."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-accounting-period-string 
 +   (N_ "End of accounting period"))
 +  (gnc:reldate-string-db 
 +   'store 'end-accounting-period-desc 
 +   (N_ "Last day of the accounting period, as set in the global preferences."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-this-month-string 
 +   (N_ "Start of this month"))
 +  (gnc:reldate-string-db 
 +   'store 'start-this-month-desc 
 +   (N_ "First day of the current month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-this-month-string 
 +   (N_ "End of this month"))
 +  (gnc:reldate-string-db 
 +   'store 'end-this-month-desc 
 +   (N_ "Last day of the current month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-month-string 
 +   (N_ "Start of previous month"))
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-month-desc
 +   (N_ "First day of the previous month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-month-string 
 +   (N_ "End of previous month"))
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-month-desc
 +   (N_ "Last day of previous month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-next-month-string 
 +   (N_ "Start of next month"))
 +  (gnc:reldate-string-db 
 +   'store 'start-next-month-desc
 +   (N_ "First day of the next month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-next-month-string 
 +   (N_ "End of next month"))
 +  (gnc:reldate-string-db 
 +   'store 'end-next-month-desc
 +   (N_ "Last day of next month."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-current-quarter-string 
 +   (N_ "Start of current quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'start-current-quarter-desc
 +   (N_ "First day of the current quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-current-quarter-string 
 +   (N_ "End of current quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'end-current-quarter-desc
 +   (N_ "Last day of the current quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-quarter-string 
 +   (N_ "Start of previous quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'start-prev-quarter-desc
 +   (N_ "First day of the previous quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-quarter-string 
 +   (N_ "End of previous quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'end-prev-quarter-desc 
 +   (N_ "Last day of previous quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'start-next-quarter-string 
 +   (N_ "Start of next quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'start-next-quarter-desc
 +   (N_ "First day of the next quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'end-next-quarter-string 
 +   (N_ "End of next quarter"))
 +  (gnc:reldate-string-db 
 +   'store 'end-next-quarter-desc 
 +   (N_ "Last day of next quarterly accounting period."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'today-string 
 +   (N_ "Today"))
 +  (gnc:reldate-string-db 
 +   'store 'today-desc (N_ "The current date."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-month-ago-string 
 +   (N_ "One Month Ago"))
 +  (gnc:reldate-string-db 
 +   'store 'one-month-ago-desc (N_ "One Month Ago."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-week-ago-string 
 +   (N_ "One Week Ago"))
 +  (gnc:reldate-string-db 
 +   'store 'one-week-ago-desc (N_ "One Week Ago."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'three-months-ago-string 
 +   (N_ "Three Months Ago"))
 +  (gnc:reldate-string-db 
 +   'store 'three-months-ago-desc (N_ "Three Months Ago."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'six-months-ago-string 
 +   (N_ "Six Months Ago"))
 +  (gnc:reldate-string-db 
 +   'store 'six-months-ago-desc (N_ "Six Months Ago."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-year-ago-string (N_ "One Year Ago"))
 +  (gnc:reldate-string-db 
 +   'store 'one-year-ago-desc (N_ "One Year Ago.")) 
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-month-ahead-string 
 +   (N_ "One Month Ahead"))
 +  (gnc:reldate-string-db 
 +   'store 'one-month-ahead-desc (N_ "One Month Ahead."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-week-ahead-string 
 +   (N_ "One Week Ahead"))
 +  (gnc:reldate-string-db 
 +   'store 'one-week-ahead-desc (N_ "One Week Ahead."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'three-months-ahead-string 
 +   (N_ "Three Months Ahead"))
 +  (gnc:reldate-string-db 
 +   'store 'three-months-ahead-desc (N_ "Three Months Ahead."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'six-months-ahead-string 
 +   (N_ "Six Months Ahead"))
 +  (gnc:reldate-string-db 
 +   'store 'six-months-ahead-desc (N_ "Six Months Ahead."))
 +
 +  (gnc:reldate-string-db 
 +   'store 'one-year-ahead-string (N_ "One Year Ahead"))
 +  (gnc:reldate-string-db 
 +   'store 'one-year-ahead-desc (N_ "One Year Ahead.")) 
 +
 +  (set! gnc:relative-date-values 
 +	(list 
 +	 (vector 'start-cal-year 
 +		 (gnc:reldate-string-db 'lookup 'start-cal-year-string)
 +		 (gnc:reldate-string-db 'lookup 'start-cal-year-desc)
 +		 gnc:get-start-cal-year)
 +	 (vector 'end-cal-year 
 +		 (gnc:reldate-string-db 'lookup 'end-cal-year-string)
 +		 (gnc:reldate-string-db 'lookup 'end-cal-year-desc)
 +		 gnc:get-end-cal-year)
 +	 (vector 'start-prev-year
 +		 (gnc:reldate-string-db 'lookup 'start-prev-year-string)
 +		 (gnc:reldate-string-db 'lookup 'start-prev-year-desc)
 +		 gnc:get-start-prev-year)
 +	 (vector 'start-next-year
 +		 (gnc:reldate-string-db 'lookup 'start-next-year-string)
 +		 (gnc:reldate-string-db 'lookup 'start-next-year-desc)
 +		 gnc:get-start-next-year)
 +	 (vector 'end-prev-year
 +		 (gnc:reldate-string-db 'lookup 'end-prev-year-string)
 +		 (gnc:reldate-string-db 'lookup 'end-prev-year-desc)
 +		 gnc:get-end-prev-year)
 +	 (vector 'end-next-year
 +		 (gnc:reldate-string-db 'lookup 'end-next-year-string)
 +		 (gnc:reldate-string-db 'lookup 'end-next-year-desc)
 +		 gnc:get-end-next-year)
 +	 (vector 'start-accounting-period
 +		 (gnc:reldate-string-db 'lookup 'start-accounting-period-string)
 +		 (gnc:reldate-string-db 'lookup 'start-accounting-period-desc)
 +		 gnc:get-start-accounting-period)
 +	 (vector 'end-accounting-period
 +		 (gnc:reldate-string-db 'lookup 'end-accounting-period-string)
 +		 (gnc:reldate-string-db 'lookup 'end-accounting-period-desc)
 +		 gnc:get-end-accounting-period)
 +	 (vector 'start-this-month
 +		 (gnc:reldate-string-db 'lookup 'start-this-month-string)
 +		 (gnc:reldate-string-db 'lookup 'start-this-month-desc)
 +		 gnc:get-start-this-month)
 +	 (vector 'end-this-month
 +		 (gnc:reldate-string-db 'lookup 'end-this-month-string)
 +		 (gnc:reldate-string-db 'lookup 'end-this-month-desc)
 +		 gnc:get-end-this-month)
 +	 (vector 'start-prev-month
 +		 (gnc:reldate-string-db 'lookup 'start-prev-month-string)
 +		 (gnc:reldate-string-db 'lookup 'start-prev-month-desc)
 +		 gnc:get-start-prev-month)
 +	 (vector 'end-prev-month
 +		 (gnc:reldate-string-db 'lookup 'end-prev-month-string)
 +		 (gnc:reldate-string-db 'lookup 'end-prev-month-desc)
 +		 gnc:get-end-prev-month)
 +	 (vector 'start-next-month
 +		 (gnc:reldate-string-db 'lookup 'start-next-month-string)
 +		 (gnc:reldate-string-db 'lookup 'start-next-month-desc)
 +		 gnc:get-start-next-month)
 +	 (vector 'end-next-month
 +		 (gnc:reldate-string-db 'lookup 'end-next-month-string)
 +		 (gnc:reldate-string-db 'lookup 'end-next-month-desc)
 +		 gnc:get-end-next-month)
 +	 (vector 'start-current-quarter
 +		 (gnc:reldate-string-db 'lookup 'start-current-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'start-current-quarter-desc)
 +		 gnc:get-start-current-quarter)
 +	 (vector 'end-current-quarter
 +		 (gnc:reldate-string-db 'lookup 'end-current-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'end-current-quarter-desc)
 +		 gnc:get-end-current-quarter)
 +	 (vector 'start-prev-quarter
 +		 (gnc:reldate-string-db 'lookup 'start-prev-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'start-prev-quarter-desc)
 +		 gnc:get-start-prev-quarter)
 +	 (vector 'end-prev-quarter
 +		 (gnc:reldate-string-db 'lookup 'end-prev-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'end-prev-quarter-desc)
 +		 gnc:get-end-prev-quarter)
 +	 (vector 'start-next-quarter
 +		 (gnc:reldate-string-db 'lookup 'start-next-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'start-next-quarter-desc)
 +		 gnc:get-start-next-quarter)
 +	 (vector 'end-next-quarter
 +		 (gnc:reldate-string-db 'lookup 'end-next-quarter-string)
 +		 (gnc:reldate-string-db 'lookup 'end-next-quarter-desc)
 +		 gnc:get-end-next-quarter)
 +	 (vector 'today
 +		 (gnc:reldate-string-db 'lookup 'today-string)
 +		 (gnc:reldate-string-db 'lookup 'today-desc)
 +		 gnc:get-today)
 +	 (vector 'one-month-ago
 +		 (gnc:reldate-string-db 'lookup 'one-month-ago-string)
 +		 (gnc:reldate-string-db 'lookup 'one-month-ago-desc)
 +		 gnc:get-one-month-ago)
 +	 (vector 'three-months-ago
 +		 (gnc:reldate-string-db 'lookup 'three-months-ago-string)
 +		 (gnc:reldate-string-db 'lookup 'three-months-ago-desc)
 +		 gnc:get-three-months-ago)
 +	 (vector 'six-months-ago
 +		 (gnc:reldate-string-db 'lookup 'six-months-ago-string)
 +		 (gnc:reldate-string-db 'lookup 'six-months-ago-desc)
 +		 gnc:get-three-months-ago)
 +	 (vector 'one-year-ago
 +		 (gnc:reldate-string-db 'lookup 'one-year-ago-string)
 +		 (gnc:reldate-string-db 'lookup 'one-year-ago-desc)
 +		 gnc:get-one-year-ago)
 +	 (vector 'one-month-ahead
 +		 (gnc:reldate-string-db 'lookup 'one-month-ahead-string)
 +		 (gnc:reldate-string-db 'lookup 'one-month-ahead-desc)
 +		 gnc:get-one-month-ahead)
 +	 (vector 'three-months-ahead
 +		 (gnc:reldate-string-db 'lookup 'three-months-ahead-string)
 +		 (gnc:reldate-string-db 'lookup 'three-months-ahead-desc)
 +		 gnc:get-three-months-ahead)
 +	 (vector 'six-months-ahead
 +		 (gnc:reldate-string-db 'lookup 'six-months-ahead-string)
 +		 (gnc:reldate-string-db 'lookup 'six-months-ahead-desc)
 +		 gnc:get-three-months-ahead)
 +	 (vector 'one-year-ahead
 +		 (gnc:reldate-string-db 'lookup 'one-year-ahead-string)
 +		 (gnc:reldate-string-db 'lookup 'one-year-ahead-desc)
 +		 gnc:get-one-year-ahead)))
 +
 +
 +  (gnc:make-reldate-hash gnc:relative-date-hash gnc:relative-date-values)
 +  (set! gnc:reldate-list
 +	(map (lambda (x) (vector-ref x 0)) gnc:relative-date-values)))

commit a0f49d8b17dede2c6f4835b700e18badf4acb025
Author: John Ralls <jralls at ceridwen.us>
Date:   Sun Mar 25 09:17:17 2018 -0700

    Revert "Load the environment file when initializing the python bindings."
    
    It's unnecessary, the swigged gnucash_core init function already does it.
    This reverts commit 7b6854c1631e27e634a6b3d4e22ee6a8389836dd.

diff --git a/bindings/python/__init__.py b/bindings/python/__init__.py
index 4082cfe..1c87473 100644
--- a/bindings/python/__init__.py
+++ b/bindings/python/__init__.py
@@ -4,7 +4,6 @@
 # instead of
 # >>> from gnucash.gnucash_core import thingy
 from gnucash.gnucash_core import *
-gnc_environment_setup()
 ##  @file
 #   @brief helper file for the importing of gnucash
 #   @author Mark Jenkins, ParIT Worker Co-operative <mark at parit.ca>

commit 9cd22ee66766f0f188becf6489cd1c87446d5262
Merge: 2b6047f a670238
Author: John Ralls <jralls at ceridwen.us>
Date:   Sun Mar 25 09:02:43 2018 -0700

    Merge Chris Lam's Fix-794030 into maint


commit a670238f80a92552f7b4771be424f7fc382df69b
Author: Christopher Lam <christopher.lck at gmail.com>
Date:   Sun Mar 25 15:47:35 2018 +0800

    Bug 794030 - relative date functions compute wrong day of month
    
    Convert > to < and rename month-length to month-days for consistency.

diff --git a/src/app-utils/date-utilities.scm b/src/app-utils/date-utilities.scm
index 72eb0f3..5ecd844 100644
--- a/src/app-utils/date-utilities.scm
+++ b/src/app-utils/date-utilities.scm
@@ -718,10 +718,10 @@
 	  (set-tm:mon now 11)
 	  (set-tm:year now (- (tm:year now) 1)))
 	(set-tm:mon now (- (tm:mon now) 1)))
-    (let ((month-length (gnc:days-in-month (+ (tm:mon now) 1)
-                                           (+ (tm:year now) 1900))))
-      (if (> month-length (tm:mday now))
-	  (set-tm:mday now month-length))
+    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
+                                         (+ (tm:year now) 1900))))
+      (if (< month-days (tm:mday now))
+	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
 
@@ -734,7 +734,7 @@
     (set:tm-mon now (- (tm:mon now) 3))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
@@ -748,7 +748,7 @@
     (set:tm-mon now (- (tm:mon now) 6))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
@@ -757,8 +757,8 @@
   (let ((now (gnc-localtime (current-time))))
     (set:tm-year now (- (tm:year now) 1))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
-                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+                                         (+ (tm:year now) 1900))))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
@@ -770,10 +770,10 @@
 	  (set-tm:mon now 0)
 	  (set-tm:year now (+ (tm:year now) 1)))
 	(set-tm:mon now (+ (tm:mon now) 1)))
-    (let ((month-length (gnc:days-in-month (+ (tm:mon now) 1)
-                                           (+ (tm:year now) 1900))))
-      (if (> month-length (tm:mday now))
-	  (set-tm:mday now month-length))
+    (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
+                                         (+ (tm:year now) 1900))))
+      (if (< month-days (tm:mday now))
+	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
 
@@ -783,10 +783,10 @@
 	(begin
 	  (set:tm-mon now (- (tm:mon now) 9))
 	  (set:tm-year now  (+ (tm:year now) 1))
-       (set:tm-mon now (+ (tm:mon now) 3))))
+          (set:tm-mon now (+ (tm:mon now) 3))))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
@@ -797,10 +797,10 @@
 	(begin
 	  (set:tm-mon now (- (tm:mon now) 6))
 	  (set:tm-year now  (+ (tm:year now) 1))
-       (set:tm-mon now (+ (tm:mon now) 6))))
+          (set:tm-mon now (+ (tm:mon now) 6))))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))
@@ -809,8 +809,8 @@
   (let ((now (gnc-localtime (current-time))))
     (set:tm-year now (+ (tm:year now) 1))
     (let ((month-days (gnc:days-in-month (+ (tm:mon now) 1)
-                                          (+ (tm:year now) 1900))))
-      (if (> month-days (tm:mday now))
+                                         (+ (tm:year now) 1900))))
+      (if (< month-days (tm:mday now))
 	  (set-tm:mday now month-days))
       (set-tm:isdst now -1)
       (gnc:date->timepair now))))

commit 2b6047f9529b6cde899e8bb208636f64e9e90ed9
Author: John Ralls <jralls at ceridwen.us>
Date:   Tue Mar 20 11:41:21 2018 -0700

    Fix a string msgid generation.
    
    Call gettext in the macro, not on the macro. xgettext doesn't run cpp and can't see inside macros.

diff --git a/src/gnome/gnc-plugin-page-account-tree.c b/src/gnome/gnc-plugin-page-account-tree.c
index ce67bd6..0d24978 100644
--- a/src/gnome/gnc-plugin-page-account-tree.c
+++ b/src/gnome/gnc-plugin-page-account-tree.c
@@ -1162,9 +1162,9 @@ gnc_plugin_page_account_tree_cmd_delete_account (GtkAction *action, GncPluginPag
     list = qof_instance_get_referring_object_list(QOF_INSTANCE(account));
     if (list != NULL)
     {
-#define EXPLANATION "The list below shows objects which make use of the account which you want to delete.\nBefore you can delete it, you must either delete those objects or else modify them so they make use\nof another account"
+#define EXPLANATION _("The list below shows objects which make use of the account which you want to delete.\nBefore you can delete it, you must either delete those objects or else modify them so they make use\nof another account")
 
-        gnc_ui_object_references_show( _(EXPLANATION), list);
+        gnc_ui_object_references_show (EXPLANATION, list);
         g_list_free(list);
         return;
     }



Summary of changes:
 bindings/python/__init__.py                  |  1 -
 gnucash/gnome/gnc-plugin-page-account-tree.c |  4 ++--
 libgnucash/app-utils/date-utilities.scm      | 36 ++++++++++++++--------------
 3 files changed, 20 insertions(+), 21 deletions(-)



More information about the gnucash-changes mailing list