summaryrefslogtreecommitdiff
path: root/walk.h
diff options
context:
space:
mode:
Diffstat (limited to 'walk.h')
-rw-r--r--walk.h284
1 files changed, 173 insertions, 111 deletions
diff --git a/walk.h b/walk.h
index 073b7c9d..862ffde0 100644
--- a/walk.h
+++ b/walk.h
@@ -20,9 +20,9 @@ struct item_handler {
template <typename T>
struct compare_items {
- const node_t * sort_order;
+ const value_expr_t * sort_order;
- compare_items(const node_t * _sort_order)
+ compare_items(const value_expr_t * _sort_order)
: sort_order(_sort_order) {
assert(sort_order);
}
@@ -40,27 +40,34 @@ struct compare_items {
//////////////////////////////////////////////////////////////////////
//
-// Several default handlers
+// Transaction handlers
//
typedef std::deque<transaction_t *> transactions_deque;
typedef std::deque<entry_t *> entries_deque;
-struct ignore_transaction : public item_handler<transaction_t>
+struct ignore_transactions : public item_handler<transaction_t>
{
virtual void operator()(transaction_t * xact) {}
};
+struct clear_display_flags : public item_handler<transaction_t>
+{
+ virtual void operator()(transaction_t * xact) {
+ xact->dflags = 0;
+ }
+};
+
class sort_transactions : public item_handler<transaction_t>
{
transactions_deque transactions;
- const node_t * sort_order;
+ const value_expr_t * sort_order;
item_handler<transaction_t> * handler;
public:
sort_transactions(item_handler<transaction_t> * _handler,
- const node_t * _sort_order)
+ const value_expr_t * _sort_order)
: sort_order(_sort_order), handler(_handler) {}
virtual ~sort_transactions() {
@@ -173,11 +180,8 @@ class collapse_transactions : public item_handler<transaction_t>
// If we've reached a new entry, report on the subtotal
// accumulated thus far.
- if (last_entry && last_entry != xact->entry) {
+ if (last_entry && last_entry != xact->entry)
report_cumulative_subtotal();
- subtotal = 0;
- count = 0;
- }
subtotal += *xact;
count++;
@@ -302,90 +306,175 @@ class interval_transactions : public subtotal_transactions
}
};
-//////////////////////////////////////////////////////////////////////
-
-#define MATCHING_TRANSACTIONS 0x01
-#define OTHER_TRANSACTIONS 0x02
-
-inline void handle_transaction(transaction_t * xact,
- item_handler<transaction_t>& handler,
- unsigned int flags)
+class related_transactions : public item_handler<transaction_t>
{
- for (transactions_list::iterator i = xact->entry->transactions.begin();
- i != xact->entry->transactions.end();
- i++)
- if ((! (flags & OTHER_TRANSACTIONS) ||
- ! ((*i)->flags & TRANSACTION_AUTO)) &&
- ! ((*i)->dflags & TRANSACTION_HANDLED) &&
- (*i == xact ?
- (flags & MATCHING_TRANSACTIONS) : (flags & OTHER_TRANSACTIONS))) {
- (*i)->dflags |= TRANSACTION_HANDLED;
- handler(*i);
- }
-}
+ bool also_matching;
-inline void walk_entries(entries_list::iterator begin,
- entries_list::iterator end,
- item_handler<transaction_t>& handler,
- const std::string& predicate,
- unsigned int flags)
-{
- item_predicate<transaction_t> pred(predicate);
+ item_handler<transaction_t> * handler;
- for (entries_list::iterator i = begin; i != end; i++)
- for (transactions_list::iterator j = (*i)->transactions.begin();
- j != (*i)->transactions.end();
- j++)
- if (pred(*j))
- handle_transaction(*j, handler, flags);
-}
+ public:
+ related_transactions(item_handler<transaction_t> * _handler,
+ bool _also_matching = false)
+ : also_matching(_also_matching), handler(_handler) {}
-inline void walk_entries(entries_list::iterator begin,
- entries_list::iterator end,
- item_handler<transaction_t>& handler)
-{
- for (entries_list::iterator i = begin; i != end; i++)
- for (transactions_list::iterator j = (*i)->transactions.begin();
- j != (*i)->transactions.end();
- j++)
- handler(*j);
-}
+ virtual ~related_transactions() {
+ handler->flush();
+ delete handler;
+ }
-struct clear_flags : public item_handler<transaction_t>
-{
virtual void operator()(transaction_t * xact) {
- xact->dflags = 0;
+ for (transactions_list::iterator i = xact->entry->transactions.begin();
+ i != xact->entry->transactions.end();
+ i++)
+ if (! ((*i)->dflags & TRANSACTION_HANDLED) &&
+ (*i == xact ? also_matching :
+ ! ((*i)->flags & TRANSACTION_AUTO))) {
+ (*i)->dflags |= TRANSACTION_HANDLED;
+ (*handler)(*i);
+ }
}
};
-inline void clear_transaction_display_flags(entries_list::iterator begin,
- entries_list::iterator end)
-{
- clear_flags handler;
- walk_entries(begin, end, handler);
-}
+//////////////////////////////////////////////////////////////////////
inline void walk_transactions(transactions_list::iterator begin,
transactions_list::iterator end,
- item_handler<transaction_t>& handler)
-{
+ item_handler<transaction_t>& handler) {
for (transactions_list::iterator i = begin; i != end; i++)
handler(*i);
}
+inline void walk_transactions(transactions_list& list,
+ item_handler<transaction_t>& handler) {
+ walk_transactions(list.begin(), list.end(), handler);
+}
+
inline void walk_transactions(transactions_deque::iterator begin,
transactions_deque::iterator end,
- item_handler<transaction_t>& handler)
-{
+ item_handler<transaction_t>& handler) {
for (transactions_deque::iterator i = begin; i != end; i++)
handler(*i);
}
+inline void walk_transactions(transactions_deque& deque,
+ item_handler<transaction_t>& handler) {
+ walk_transactions(deque.begin(), deque.end(), handler);
+}
+
+inline void walk_entries(entries_list::iterator begin,
+ entries_list::iterator end,
+ item_handler<transaction_t>& handler) {
+ // jww (2004-08-11): do transaction dflags need to be cleared first?
+ for (entries_list::iterator i = begin; i != end; i++)
+ walk_transactions((*i)->transactions, handler);
+}
+
+inline void walk_entries(entries_list& list,
+ item_handler<transaction_t>& handler) {
+ walk_entries(list.begin(), list.end(), handler);
+}
+
+
+//////////////////////////////////////////////////////////////////////
+//
+// Account handlers
+//
+
typedef std::deque<account_t *> accounts_deque;
+struct add_to_account_value : public item_handler<transaction_t>
+{
+ virtual void operator()(transaction_t * xact) {
+ xact->account->value += *xact;
+ }
+};
+
+#if 0
+
+class format_accounts : public item_handler<account_t>
+{
+};
+
+class filter_accounts : public item_handler<account_t>
+{
+ item_handler<account_t> * handler;
+
+ public:
+ filter_accounts(item_handler<account_t> * _handler)
+ : handler(_handler) {}
+
+ virtual ~filter_accounts() {
+ handler->flush();
+ delete handler;
+ }
+
+ virtual void flush() {}
+
+ virtual void operator()(account_t * account) {
+ }
+};
+
+class sort_accounts : public item_handler<account_t>
+{
+ value_expr_t * sort_order;
+
+ item_handler<account_t> * handler;
+
+ public:
+ sort_accounts(item_handler<account_t> * _handler,
+ value_expr_t * _sort_order)
+ : sort_order(_sort_order), handler(_handler) {}
+
+ virtual ~sort_accounts() {
+ handler->flush();
+ delete handler;
+ }
+
+ virtual void flush() {}
+
+ virtual void operator()(account_t * account) {
+ accounts_deque accounts;
+
+ for (accounts_map::iterator i = account->accounts.begin();
+ i != account->accounts.end();
+ i++)
+ accounts.push_back((*i).second);
+
+ std::stable_sort(accounts.begin(), accounts.end(),
+ compare_items<account_t>(sort_order));
+ }
+};
+
+class balance_accounts : public item_handler<account_t>
+{
+ item_handler<account_t> * handler;
+
+ public:
+ balance_accounts(item_handler<account_t> * _handler)
+ : handler(_handler) {}
+
+ virtual ~balance_accounts() {
+ handler->flush();
+ delete handler;
+ }
+
+ virtual void flush() {
+ if (format_account::disp_subaccounts_p(top)) {
+ std::string end_format = "--------------------\n";
+ format.reset(end_format + f);
+ format.format_elements(std::cout, details_t(top));
+ }
+ }
+
+ virtual void operator()(account_t * account) {
+ }
+};
+
+#endif
+
inline void sort_accounts(account_t * account,
accounts_deque& accounts,
- const node_t * sort_order)
+ const value_expr_t * sort_order)
{
for (accounts_map::iterator i = account->accounts.begin();
i != account->accounts.end();
@@ -396,32 +485,14 @@ inline void sort_accounts(account_t * account,
compare_items<account_t>(sort_order));
}
-inline void walk__accounts(account_t * account,
- item_handler<account_t>& handler)
-{
- handler(account);
-
- for (accounts_map::const_iterator i = account->accounts.begin();
- i != account->accounts.end();
- i++)
- walk__accounts((*i).second, handler);
-}
-
inline void walk__accounts_sorted(account_t * account,
item_handler<account_t>& handler,
- const node_t * sort_order)
+ const value_expr_t * sort_order)
{
handler(account);
accounts_deque accounts;
-
- for (accounts_map::const_iterator i = account->accounts.begin();
- i != account->accounts.end();
- i++)
- accounts.push_back((*i).second);
-
- std::stable_sort(accounts.begin(), accounts.end(),
- compare_items<account_t>(sort_order));
+ sort_accounts(account, accounts, sort_order);
for (accounts_deque::const_iterator i = accounts.begin();
i != accounts.end();
@@ -429,21 +500,6 @@ inline void walk__accounts_sorted(account_t * account,
walk__accounts_sorted(*i, handler, sort_order);
}
-inline void for_each_account(account_t * account,
- item_handler<account_t>& handler)
-{
- handler(account);
-
- for (accounts_map::iterator i = account->accounts.begin();
- i != account->accounts.end();
- i++)
- walk__accounts((*i).second, handler);
-}
-
-void calc__accounts(account_t * account,
- const item_predicate<transaction_t>& pred,
- unsigned int flags);
-
inline void sum__accounts(account_t * account)
{
for (accounts_map::iterator i = account->accounts.begin();
@@ -455,16 +511,22 @@ inline void sum__accounts(account_t * account)
account->total += account->value;
}
-inline void walk_accounts(account_t * account,
- item_handler<account_t>& handler,
- const std::string& predicate,
- unsigned int flags,
- const bool calc_subtotals,
- const node_t * sort_order = NULL)
+inline void walk__accounts(account_t * account,
+ item_handler<account_t>& handler)
{
- item_predicate<transaction_t> pred(predicate);
+ handler(account);
- calc__accounts(account, pred, flags);
+ for (accounts_map::const_iterator i = account->accounts.begin();
+ i != account->accounts.end();
+ i++)
+ walk__accounts((*i).second, handler);
+}
+
+inline void walk_accounts(account_t * account,
+ item_handler<account_t>& handler,
+ const bool calc_subtotals,
+ const value_expr_t * sort_order = NULL)
+{
if (calc_subtotals)
sum__accounts(account);