#ifndef _WALK_H #define _WALK_H #include "ledger.h" #include "balance.h" #include "format.h" #include "valexpr.h" #include #include namespace ledger { template class item_predicate { const node_t * predicate; public: item_predicate(const node_t * _predicate) : predicate(_predicate) {} bool operator()(const T * item) const { if (predicate) { balance_t result; predicate->compute(result, details_t(item)); return result; } else { return true; } } }; inline void add_to_balance_pair(balance_pair_t& balance, transaction_t * xact, const bool inverted = false) { if (inverted) { balance.quantity += - xact->amount; balance.cost += - xact->cost; } else { balance += *xact; } } class format_transaction { std::ostream& output_stream; const format_t& first_line_format; const format_t& next_lines_format; const bool inverted; unsigned int index; entry_t * last_entry; public: format_transaction(std::ostream& _output_stream, const format_t& _first_line_format, const format_t& _next_lines_format, const bool _inverted) : output_stream(_output_stream), first_line_format(_first_line_format), next_lines_format(_next_lines_format), inverted(_inverted), index(0), last_entry(NULL) {} void operator()(transaction_t * xact); }; template struct compare_items { const node_t * sort_order; compare_items(const node_t * _sort_order) : sort_order(_sort_order) { assert(sort_order); } bool operator()(const T * left, const T * right) const { assert(left); assert(right); balance_t left_result; sort_order->compute(left_result, details_t(left)); balance_t right_result; sort_order->compute(right_result, details_t(right)); return left_result < right_result; } }; typedef std::deque transactions_deque; class collect_transactions { transactions_deque& transactions; public: collect_transactions(transactions_deque& _transactions) : transactions(_transactions) {} void operator()(transaction_t * xact) { transactions.push_back(xact); } }; inline void sort_transactions(transactions_deque& transactions, const node_t * sort_order) { std::stable_sort(transactions.begin(), transactions.end(), compare_items(sort_order)); } class ignore_transaction { public: void operator()(transaction_t * xact) const {} }; #define MATCHING_TRANSACTIONS 0x01 #define OTHER_TRANSACTIONS 0x02 template void handle_transaction(transaction_t * xact, Function functor, item_predicate& pred_functor, unsigned int flags) { if ((flags & MATCHING_TRANSACTIONS) && ! (xact->flags & TRANSACTION_HANDLED)) { xact->flags |= TRANSACTION_HANDLED; if (pred_functor(xact)) { xact->flags |= TRANSACTION_DISPLAYED; functor(xact); } } if (flags & OTHER_TRANSACTIONS) for (transactions_list::iterator i = xact->entry->transactions.begin(); i != xact->entry->transactions.end(); i++) { if (*i == xact || ((*i)->flags & (TRANSACTION_AUTO | TRANSACTION_HANDLED))) continue; (*i)->flags |= TRANSACTION_HANDLED; if (pred_functor(xact)) { xact->flags |= TRANSACTION_DISPLAYED; functor(*i); } } } template void walk_entries(entries_list::iterator begin, entries_list::iterator end, Function functor, const node_t * predicate, unsigned int flags, const node_t * display_predicate = NULL) { item_predicate pred_functor(predicate); item_predicate disp_pred_functor(display_predicate); for (entries_list::iterator i = begin; i != end; i++) for (transactions_list::iterator j = (*i)->transactions.begin(); j != (*i)->transactions.end(); j++) if (pred_functor(*j)) handle_transaction(*j, functor, disp_pred_functor, flags); } template void walk_entries(entries_list::iterator begin, entries_list::iterator end, Function functor) { for (entries_list::iterator i = begin; i != end; i++) for (transactions_list::iterator j = (*i)->transactions.begin(); j != (*i)->transactions.end(); j++) functor(*j); } class clear_flags { public: void operator()(transaction_t * xact) const { xact->flags &= ~(TRANSACTION_HANDLED | TRANSACTION_DISPLAYED); } }; inline void clear_transaction_display_flags(entries_list::iterator begin, entries_list::iterator end) { walk_entries(begin, end, clear_flags()); } template void walk_transactions(transactions_list::iterator begin, transactions_list::iterator end, Function functor) { for (transactions_list::iterator i = begin; i != end; i++) functor(*i); } template void walk_transactions(transactions_deque::iterator begin, transactions_deque::iterator end, Function functor) { for (transactions_deque::iterator i = begin; i != end; i++) functor(*i); } class format_account { std::ostream& output_stream; const format_t& format; const account_t * last_account; public: format_account(std::ostream& _output_stream, const format_t& _format) : output_stream(_output_stream), format(_format) {} void operator()(const account_t * account, bool report_top = false); }; typedef std::deque accounts_deque; inline void sort_accounts(account_t * account, accounts_deque& accounts, const node_t * sort_order) { 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(sort_order)); } template void walk__accounts(const account_t * account, Function functor, item_predicate& disp_pred_functor) { if (disp_pred_functor(account)) functor(account); for (accounts_map::const_iterator i = account->accounts.begin(); i != account->accounts.end(); i++) walk__accounts((*i).second, functor, disp_pred_functor); } template void walk__accounts_sorted(const account_t * account, Function functor, const node_t * sort_order, item_predicate& disp_pred_functor) { if (disp_pred_functor(account)) functor(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(sort_order)); for (accounts_deque::const_iterator i = accounts.begin(); i != accounts.end(); i++) walk__accounts_sorted(*i, functor, sort_order, disp_pred_functor); } template void for_each_account(account_t * account, Function functor) { functor(account); for (accounts_map::iterator i = account->accounts.begin(); i != account->accounts.end(); i++) walk__accounts((*i).second, functor); } void calc__accounts(account_t * account, item_predicate& pred_functor, unsigned int flags); inline void sum__accounts(account_t * account) { for (accounts_map::iterator i = account->accounts.begin(); i != account->accounts.end(); i++) { sum__accounts((*i).second); account->total += (*i).second->total; } account->total += account->value; } template void walk_accounts(account_t * account, Function functor, const node_t * predicate, unsigned int flags, const bool calc_subtotals, const node_t * display_predicate = NULL, const node_t * sort_order = NULL) { item_predicate pred_functor(predicate); item_predicate disp_pred_functor(display_predicate); calc__accounts(account, pred_functor, flags); if (calc_subtotals) sum__accounts(account); if (sort_order) walk__accounts_sorted(account, functor, sort_order, disp_pred_functor); else walk__accounts(account, functor, disp_pred_functor); } } // namespace ledger #endif // _WALK_H