summaryrefslogtreecommitdiff
path: root/src/entry.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/entry.cc')
-rw-r--r--src/entry.cc103
1 files changed, 75 insertions, 28 deletions
diff --git a/src/entry.cc b/src/entry.cc
index a07481f1..eceec5f7 100644
--- a/src/entry.cc
+++ b/src/entry.cc
@@ -85,9 +85,6 @@ bool entry_base_t::finalize()
// for auto-calculating the value of entries with no cost, and the per-unit
// price of unpriced commodities.
- // (let ((balance 0)
- // null-xact)
-
value_t balance;
xact_t * null_xact = NULL;
@@ -119,8 +116,8 @@ bool entry_base_t::finalize()
DEBUG("entry.finalize", "initial balance = " << balance);
- // If there is only one xact, balance against the default account if
- // one has been set.
+ // If there is only one xact, balance against the default account if one has
+ // been set.
if (journal && journal->basket && xacts.size() == 1 && ! balance.is_null()) {
// jww (2008-07-24): Need to make the rest of the code aware of what to do
@@ -131,9 +128,9 @@ bool entry_base_t::finalize()
}
if (null_xact != NULL) {
- // If one xact has no value at all, its value will become the
- // inverse of the rest. If multiple commodities are involved, multiple
- // xacts are generated to balance them all.
+ // If one xact has no value at all, its value will become the inverse of
+ // the rest. If multiple commodities are involved, multiple xacts are
+ // generated to balance them all.
if (balance.is_balance()) {
bool first = true;
@@ -161,41 +158,88 @@ bool entry_base_t::finalize()
else if (balance.is_balance() &&
balance.as_balance().amounts.size() == 2) {
// When an entry involves two different commodities (regardless of how
- // many xacts there are) determine the conversion ratio by dividing
- // the total value of one commodity by the total value of the other. This
- // establishes the per-unit cost for this xact for both
- // commodities.
+ // many xacts there are) determine the conversion ratio by dividing the
+ // total value of one commodity by the total value of the other. This
+ // establishes the per-unit cost for this xact for both commodities.
+
+ DEBUG("entry.finalize", "there were exactly two commodities");
+
+ bool saw_cost = false;
+ xact_t * top_xact = NULL;
+
+ foreach (xact_t * xact, xacts) {
+ if (! xact->amount.is_null())
+ if (xact->amount.is_annotated())
+ top_xact = xact;
+ else if (! top_xact)
+ top_xact = xact;
+
+ if (xact->cost) {
+ saw_cost = true;
+ break;
+ }
+ }
+
+ if (! saw_cost && top_xact) {
+ const balance_t& bal(balance.as_balance());
- const balance_t& bal(balance.as_balance());
+ DEBUG("entry.finalize", "there were no costs, and a valid top_xact");
- balance_t::amounts_map::const_iterator a = bal.amounts.begin();
+ balance_t::amounts_map::const_iterator a = bal.amounts.begin();
- const amount_t& x((*a++).second);
- const amount_t& y((*a++).second);
+ const amount_t * x = &(*a++).second;
+ const amount_t * y = &(*a++).second;
- if (! y.is_realzero()) {
- amount_t per_unit_cost = (x / y).abs();
+ if (x->commodity() != top_xact->amount.commodity()) {
+ const amount_t * t = x;
+ x = y;
+ y = t;
+ }
+
+ DEBUG("entry.finalize", "primary amount = " << *y);
+ DEBUG("entry.finalize", "secondary amount = " << *x);
- commodity_t& comm(x.commodity());
+ commodity_t& comm(x->commodity());
+ amount_t per_unit_cost;
+ amount_t total_cost;
+
+ foreach (xact_t * xact, xacts) {
+ if (xact != top_xact && xact->must_balance() &&
+ ! xact->amount.is_null() &&
+ xact->amount.is_annotated() &&
+ xact->amount.annotation().price) {
+ amount_t temp = *xact->amount.annotation().price * xact->amount;
+ if (total_cost.is_null()) {
+ total_cost = temp;
+ y = &total_cost;
+ } else {
+ total_cost += temp;
+ }
+ DEBUG("entry.finalize", "total_cost = " << total_cost);
+ }
+ }
+ per_unit_cost = (*y / *x).abs();
+
+ DEBUG("entry.finalize", "per_unit_cost = " << per_unit_cost);
foreach (xact_t * xact, xacts) {
const amount_t& amt(xact->amount);
- if (! (xact->cost || ! xact->must_balance() ||
- amt.commodity() == comm)) {
+ if (xact->must_balance() && amt.commodity() == comm) {
balance -= amt;
xact->cost = per_unit_cost * amt;
balance += *xact->cost;
- }
+ DEBUG("entry.finalize", "set xact->cost to = " << *xact->cost);
+ }
}
}
DEBUG("entry.finalize", "resolved balance = " << balance);
}
- // Now that the xact list has its final form, calculate the balance
- // once more in terms of total cost, accounting for any possible gain/loss
+ // Now that the xact list has its final form, calculate the balance once
+ // more in terms of total cost, accounting for any possible gain/loss
// amounts.
foreach (xact_t * xact, xacts) {
@@ -204,9 +248,12 @@ bool entry_base_t::finalize()
throw_(balance_error, "Transaction's cost must be of a different commodity");
commodity_t::cost_breakdown_t breakdown =
- commodity_t::exchange(xact->amount, *xact->cost);
+ commodity_t::exchange(xact->amount, *xact->cost, false,
+ datetime_t(date(), time_duration(0, 0, 0, 0)));
- if (xact->amount.is_annotated())
+ if (xact->amount.is_annotated() &&
+ breakdown.basis_cost.commodity() ==
+ breakdown.final_cost.commodity())
add_or_set_value(balance, (breakdown.basis_cost -
breakdown.final_cost).rounded());
else
@@ -231,8 +278,8 @@ bool entry_base_t::finalize()
if (! xact->amount.is_null()) {
all_null = false;
- // jww (2008-08-09): For now, this feature only works for
- // non-specific commodities.
+ // jww (2008-08-09): For now, this feature only works for non-specific
+ // commodities.
add_or_set_value(xact->account->xdata().value, xact->amount);
DEBUG("entry.finalize.totals",