summaryrefslogtreecommitdiff
path: root/src/print.cc
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2010-03-07 22:47:07 -0500
committerJohn Wiegley <johnw@newartisans.com>2010-03-08 01:11:48 -0500
commit75b7294a6db10e7f7b18b6aeef1aa0f568124a1d (patch)
tree0fecbc12acaacb33cf3c241c536de3347d2742ed /src/print.cc
parent7e79cd82cd4ad5f87f40e6beebdb53e65bb11168 (diff)
downloadfork-ledger-75b7294a6db10e7f7b18b6aeef1aa0f568124a1d.tar.gz
fork-ledger-75b7294a6db10e7f7b18b6aeef1aa0f568124a1d.tar.bz2
fork-ledger-75b7294a6db10e7f7b18b6aeef1aa0f568124a1d.zip
Rewrite the "print" command as a custom function
There ended up being too many corner cases for the generalized formatter to handle.
Diffstat (limited to 'src/print.cc')
-rw-r--r--src/print.cc208
1 files changed, 208 insertions, 0 deletions
diff --git a/src/print.cc b/src/print.cc
new file mode 100644
index 00000000..e190ad31
--- /dev/null
+++ b/src/print.cc
@@ -0,0 +1,208 @@
+/*
+ * Copyright (c) 2003-2010, John Wiegley. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are
+ * met:
+ *
+ * - Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * - Neither the name of New Artisans LLC nor the names of its
+ * contributors may be used to endorse or promote products derived from
+ * this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <system.hh>
+
+#include "print.h"
+#include "xact.h"
+#include "post.h"
+#include "account.h"
+#include "session.h"
+#include "report.h"
+
+namespace ledger {
+
+print_xacts::print_xacts(report_t& _report,
+ bool _print_raw)
+ : report(_report), print_raw(_print_raw)
+{
+ TRACE_CTOR(print_xacts, "report&, bool");
+}
+
+namespace {
+ void print_note(std::ostream& out, const string& note)
+ {
+ if (note.length() > 15)
+ out << "\n ;";
+ else
+ out << " ;";
+
+ bool need_separator = false;
+ for (const char * p = note.c_str(); *p; p++) {
+ if (*p == '\n') {
+ need_separator = true;
+ } else {
+ if (need_separator) {
+ out << "\n ;";
+ need_separator = false;
+ }
+ out << *p;
+ }
+ }
+ }
+
+ void print_xact(report_t& report, std::ostream& out, xact_t& xact)
+ {
+ // jww (2010-03-07): Also add support for --raw
+
+ out << format_date(item_t::use_effective_date ?
+ xact.date() : xact.actual_date(),
+ FMT_WRITTEN);
+ if (! item_t::use_effective_date && xact.effective_date())
+ out << '=' << format_date(*xact.effective_date(), FMT_WRITTEN);
+ out << ' ';
+
+ // jww (2010-03-06): Output posting state, if different
+ out << (xact.state() == item_t::CLEARED ? "* " :
+ (xact.state() == item_t::PENDING ? "! " : ""));
+
+ if (xact.code)
+ out << '(' << *xact.code << ") ";
+
+ out << xact.payee;
+
+ if (xact.note)
+ print_note(out, *xact.note);
+ out << '\n';
+
+ if (xact.metadata) {
+ foreach (const item_t::string_map::value_type& data, *xact.metadata) {
+ if (! data.second.second) {
+ out << " ; ";
+ if (data.second.first)
+ out << data.first << ": " << *data.second.first;
+ else
+ out << ':' << data.first << ":";
+ out << '\n';
+ }
+ }
+ }
+
+ foreach (post_t * post, xact.posts) {
+ if (post->has_flags(ITEM_TEMP | ITEM_GENERATED) &&
+ ! report.HANDLED(print_virtual))
+ continue;
+
+ out << " ";
+ // jww (2010-03-06): Output posting state, if different
+
+ std::ostringstream buf;
+
+ if (post->has_flags(POST_VIRTUAL)) {
+ if (post->has_flags(POST_MUST_BALANCE))
+ buf << '[';
+ else
+ buf << '(';
+ }
+
+ buf << post->account->fullname();
+
+ if (post->has_flags(POST_VIRTUAL)) {
+ if (post->has_flags(POST_MUST_BALANCE))
+ buf << ']';
+ else
+ buf << ')';
+ }
+
+ if (! post->has_flags(POST_CALCULATED) || report.HANDLED(print_virtual)) {
+ unistring name(buf.str());
+
+ out << name.extract();
+ int slip = 36 - static_cast<int>(name.length());
+ if (slip > 0)
+ out << string(slip, ' ');
+
+ std::ostringstream amt_str;
+ report.scrub(post->amount).print(amt_str, 12, -1, true);
+ string amt = amt_str.str();
+ string trimmed_amt(amt);
+ trim_left(trimmed_amt);
+ int amt_slip = (static_cast<int>(amt.length()) -
+ static_cast<int>(trimmed_amt.length()));
+ if (slip + amt_slip < 2)
+ out << string(2 - (slip + amt_slip), ' ');
+ out << amt;
+
+ if (post->cost && ! post->has_flags(POST_CALCULATED)) {
+ if (post->has_flags(POST_COST_IN_FULL))
+ out << " @@ " << report.scrub(post->cost->abs());
+ else
+ out << " @ " << report.scrub((*post->cost / post->amount).abs());
+ }
+
+ if (post->assigned_amount)
+ out << " = " << report.scrub(*post->assigned_amount);
+ } else {
+ out << buf.str();
+ }
+
+ if (post->note)
+ print_note(out, *post->note);
+ out << '\n';
+ }
+ }
+}
+
+void print_xacts::flush()
+{
+ std::ostream& out(report.output_stream);
+
+ bool first = true;
+ foreach (xact_t * xact, xacts) {
+ if (first)
+ first = false;
+ else
+ out << '\n';
+
+ if (print_raw) {
+ print_item(out, *xact);
+ out << '\n';
+ } else {
+ print_xact(report, out, *xact);
+ }
+ }
+
+ out.flush();
+}
+
+void print_xacts::operator()(post_t& post)
+{
+ if (! post.has_xdata() ||
+ ! post.xdata().has_flags(POST_EXT_DISPLAYED)) {
+ if (xacts_present.find(post.xact) == xacts_present.end()) {
+ xacts_present.insert(xacts_present_map::value_type(post.xact, true));
+ xacts.push_back(post.xact);
+ }
+ post.xdata().add_flags(POST_EXT_DISPLAYED);
+ }
+}
+
+} // namespace ledger