#ifndef _FORMAT_H #define _FORMAT_H #include "ledger.h" #include "valexpr.h" #include "walk.h" namespace ledger { std::string truncated(const std::string& str, unsigned int width); std::string partial_account_name(const account_t * account, const unsigned int start_depth); struct element_t { enum kind_t { STRING, VALUE_EXPR, DATE_STRING, CLEARED, CODE, PAYEE, ACCOUNT_NAME, ACCOUNT_FULLNAME, OPT_AMOUNT, VALUE, TOTAL, SPACER }; bool align_left; unsigned int min_width; unsigned int max_width; kind_t type; std::string chars; node_t * val_expr; struct element_t * next; element_t() : align_left(false), min_width(0), max_width(0), type(STRING), val_expr(NULL), next(NULL) {} ~element_t() { if (val_expr) delete val_expr; if (next) delete next; // recursive, but not too deep } }; struct format_t { element_t * elements; static std::auto_ptr value_expr; static std::auto_ptr total_expr; format_t(const std::string& _format) : elements(NULL) { reset(_format); } ~format_t() { if (elements) delete elements; } void reset(const std::string& _format) { if (elements) delete elements; elements = parse_elements(_format); } static element_t * parse_elements(const std::string& fmt); void format_elements(std::ostream& out, const details_t& details) const; static void compute_value(balance_t& result, const details_t& details) { if (value_expr.get()) value_expr->compute(result, details); } static void compute_total(balance_t& result, const details_t& details) { if (total_expr.get()) total_expr->compute(result, details); } }; class format_transactions : public item_handler { std::ostream& output_stream; const format_t& first_line_format; const format_t& next_lines_format; entry_t * last_entry; public: format_transactions(std::ostream& _output_stream, const format_t& _first_line_format, const format_t& _next_lines_format) : output_stream(_output_stream), first_line_format(_first_line_format), next_lines_format(_next_lines_format), last_entry(NULL) {} virtual ~format_transactions() { flush(); } virtual void flush() { output_stream.flush(); } virtual void operator()(transaction_t * xact) { xact->dflags |= TRANSACTION_DISPLAYED; // This makes the assumption that transactions from a single entry // are always grouped together. if (last_entry != xact->entry) first_line_format.format_elements(output_stream, details_t(xact)); else next_lines_format.format_elements(output_stream, details_t(xact)); last_entry = xact->entry; } }; class format_account : public item_handler { std::ostream& output_stream; const format_t& format; item_predicate disp_pred; public: format_account(std::ostream& _output_stream, const format_t& _format, const std::string& display_predicate = NULL) : output_stream(_output_stream), format(_format), disp_pred(display_predicate) {} virtual ~format_account() { flush(); } static bool disp_subaccounts_p(const account_t * account, const item_predicate& disp_pred, const account_t *& to_show); static bool disp_subaccounts_p(const account_t * account) { const account_t * temp; return disp_subaccounts_p(account, item_predicate(NULL), temp); } static bool display_account(const account_t * account, const item_predicate& disp_pred); virtual void flush() { output_stream.flush(); } virtual void operator()(account_t * account) { if (display_account(account, disp_pred)) { format.format_elements(output_stream, details_t(account)); account->dflags |= ACCOUNT_DISPLAYED; } } }; class format_equity : public item_handler { std::ostream& output_stream; const format_t& first_line_format; const format_t& next_lines_format; item_predicate disp_pred; mutable balance_t total; public: format_equity(std::ostream& _output_stream, const format_t& _first_line_format, const format_t& _next_lines_format, const std::string& display_predicate = NULL) : output_stream(_output_stream), first_line_format(_first_line_format), next_lines_format(_next_lines_format), disp_pred(display_predicate) { entry_t header_entry; header_entry.payee = "Opening Balances"; header_entry.date = std::time(NULL); first_line_format.format_elements(output_stream, details_t(&header_entry)); } virtual ~format_equity() { flush(); } virtual void flush() { account_t summary(NULL, "Equity:Opening Balances"); summary.value = - total; next_lines_format.format_elements(output_stream, details_t(&summary)); output_stream.flush(); } virtual void operator()(account_t * account) { if (format_account::display_account(account, disp_pred)) { next_lines_format.format_elements(output_stream, details_t(account)); account->dflags |= ACCOUNT_DISPLAYED; total += account->value.quantity; } } }; } // namespace ledger #endif // _REPORT_H