summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/numerics/balance.h20
-rw-r--r--src/numerics/balpair.h249
2 files changed, 154 insertions, 115 deletions
diff --git a/src/numerics/balance.h b/src/numerics/balance.h
index b132115f..34b2fcc7 100644
--- a/src/numerics/balance.h
+++ b/src/numerics/balance.h
@@ -77,14 +77,12 @@ class balance_t
public:
typedef std::map<const commodity_t *, amount_t> amounts_map;
-protected:
amounts_map amounts;
// jww (2007-05-20): Remove these two by adding access methods
friend class value_t;
friend class entry_base_t;
-public:
/**
* Constructors. balance_t supports similar forms of construction
* to amount_t.
@@ -233,7 +231,8 @@ public:
balance_t& operator-=(const balance_t& bal);
balance_t& operator-=(const amount_t& amt);
- balance_t& operator*=(const amount_t& amt);
+ virtual balance_t& operator*=(const amount_t& amt);
+
balance_t& operator*=(const double val) {
return *this *= amount_t(val);
}
@@ -244,7 +243,8 @@ public:
return *this *= amount_t(val);
}
- balance_t& operator/=(const amount_t& amt);
+ virtual balance_t& operator/=(const amount_t& amt);
+
balance_t& operator/=(const double val) {
return *this /= amount_t(val);
}
@@ -294,7 +294,7 @@ public:
temp.in_place_negate();
return temp;
}
- balance_t& in_place_negate() {
+ virtual balance_t& in_place_negate() {
for (amounts_map::iterator i = amounts.begin();
i != amounts.end();
i++)
@@ -319,10 +319,10 @@ public:
temp.in_place_reduce();
return temp;
}
- balance_t& in_place_reduce() {
- balance_t temp;
+ virtual balance_t& in_place_reduce() {
// A temporary must be used here because reduction may cause
// multiple component amounts to collapse to the same commodity.
+ balance_t temp;
for (amounts_map::const_iterator i = amounts.begin();
i != amounts.end();
i++)
@@ -335,10 +335,10 @@ public:
temp.in_place_unreduce();
return temp;
}
- balance_t& in_place_unreduce() {
- balance_t temp;
+ virtual balance_t& in_place_unreduce() {
// A temporary must be used here because unreduction may cause
// multiple component amounts to collapse to the same commodity.
+ balance_t temp;
for (amounts_map::const_iterator i = amounts.begin();
i != amounts.end();
i++)
@@ -499,7 +499,7 @@ public:
out << ")";
}
- bool valid() const {
+ virtual bool valid() const {
for (amounts_map::const_iterator i = amounts.begin();
i != amounts.end();
i++)
diff --git a/src/numerics/balpair.h b/src/numerics/balpair.h
index 4aa682fc..96ccf42a 100644
--- a/src/numerics/balpair.h
+++ b/src/numerics/balpair.h
@@ -97,9 +97,18 @@ public:
balance_pair_t(const balance_t& bal) : balance_t(bal) {
TRACE_CTOR(balance_pair_t, "const balance_t&");
}
+ balance_pair_t(const balance_t& bal,
+ const balance_t& cost_bal)
+ : balance_t(bal), cost(cost_bal) {
+ TRACE_CTOR(balance_pair_t, "const balance_t&, const balance_t&");
+ }
balance_pair_t(const amount_t& amt) : balance_t(amt) {
TRACE_CTOR(balance_pair_t, "const amount_t&");
}
+ balance_pair_t(const amount_t& amt, const amount_t& cost_amt)
+ : balance_t(amt), cost(cost_amt) {
+ TRACE_CTOR(balance_pair_t, "const amount_t&, const amount_t&");
+ }
balance_pair_t(const double val) : balance_t(val) {
TRACE_CTOR(balance_pair_t, "const double");
}
@@ -161,12 +170,17 @@ public:
* subtraction of other balance pairs, balances or amounts, but
* multiplication and division are restricted to uncommoditized
* amounts only.
+ *
+ * There is also an additional additive method called `add' which
+ * allows for adding an amount and an associated cost
+ * simultaneously. The signature is:
+ * add(amount_t amount, optional<amount_t> cost)
*/
balance_pair_t& operator+=(const balance_pair_t& bal_pair) {
balance_t::operator+=(bal_pair);
if (bal_pair.cost) {
if (! cost)
- *cost = quantity();
+ cost = quantity();
*cost += *bal_pair.cost;
}
return *this;
@@ -175,153 +189,178 @@ public:
balance_t::operator+=(bal_pair);
if (bal_pair.cost) {
if (! cost)
- *cost = quantity();
+ cost = quantity();
*cost += *bal_pair.cost;
}
return *this;
}
- balance_pair_t& operator*=(const amount_t& amt) {
+ virtual balance_pair_t& operator*=(const amount_t& amt) {
balance_t::operator*=(amt);
if (cost)
*cost *= amt;
return *this;
}
- balance_pair_t& operator*=(const double val) {
- return *this *= amount_t(val);
- }
- balance_pair_t& operator*=(const unsigned long val) {
- return *this *= amount_t(val);
- }
- balance_pair_t& operator*=(const long val) {
- return *this *= amount_t(val);
- }
- balance_pair_t& operator/=(const amount_t& amt) {
+
+ virtual balance_pair_t& operator/=(const amount_t& amt) {
balance_t::operator/=(amt);
if (cost)
*cost /= amt;
return *this;
}
- // comparison
- bool operator==(const balance_pair_t& bal_pair) const {
- return quantity() == bal_pair.quantity();
- }
- bool operator==(const balance_t& bal) const {
- return quantity() == bal;
- }
- bool operator==(const amount_t& amt) const {
- return quantity() == amt;
+ balance_pair_t& add(const amount_t& amt,
+ const optional<amount_t>& a_cost = none) {
+ if (a_cost && ! cost)
+ cost = quantity();
+
+ *this += amt;
+
+ if (cost)
+ *cost += a_cost ? *a_cost : amt;
+
+ return *this;
}
/**
- * The `quantity' method provides direct access to the balance_t
- * base-class part of the balance pair.
+ * Unary arithmetic operators. There are only a few unary methods
+ * supported for balance pairs (otherwise, the operators inherited
+ * from balance_t are used):
+ *
+ * abs() returns the absolute value of both the quantity and the
+ * cost of a balance pair.
+ *
+ * in_place_negate() negates all the amounts in both the quantity
+ * and the cost.
+ *
+ * in_place_reduce() reduces all the amounts in both the quantity
+ * and the cost.
+ *
+ * in_place_unreduce() unreduces all the amounts in both the
+ * quantity and the cost.
+ *
+ * quantity() returns the balance part of a balance. It is the same
+ * as doing a downcast<balance_t>(balance_pair).
*/
- balance_t& quantity() {
- return *this;
- }
- const balance_t& quantity() const {
- return *this;
+ balance_pair_t abs() const {
+ balance_t temp;
+ for (amounts_map::const_iterator i = amounts.begin();
+ i != amounts.end();
+ i++)
+ temp += i->second.abs();
+
+ if (cost) {
+ balance_t cost_temp;
+ for (amounts_map::const_iterator i = cost->amounts.begin();
+ i != cost->amounts.end();
+ i++)
+ cost_temp += i->second.abs();
+ return balance_pair_t(temp, cost_temp);
+ }
+ return temp;
}
- // unary negation
- void in_place_negate() {
-#if 0
- quantity.in_place_negate();
+ virtual balance_t& in_place_negate() {
+ balance_t::in_place_negate();
if (cost)
cost->in_place_negate();
-#endif
- }
- balance_pair_t negate() const {
- balance_pair_t temp = *this;
- temp.in_place_negate();
- return temp;
- }
- balance_pair_t operator-() const {
- return negate();
+ return *this;
}
- // test for non-zero (use ! for zero)
- operator bool() const {
- return quantity();
- }
+ virtual balance_t& in_place_reduce() {
+ // A temporary must be used here because reduction may cause
+ // multiple component amounts to collapse to the same commodity.
+ balance_t temp;
+ for (amounts_map::const_iterator i = amounts.begin();
+ i != amounts.end();
+ i++)
+ temp += i->second.reduce();
- bool is_realzero() const {
-#if 0
- return ((! cost || cost->is_realzero()) && quantity.is_realzero());
-#else
- return false;
-#endif
+ if (cost) {
+ balance_t cost_temp;
+ for (amounts_map::const_iterator i = cost->amounts.begin();
+ i != cost->amounts.end();
+ i++)
+ cost_temp += i->second.reduce();
+ return *this = balance_pair_t(temp, cost_temp);
+ }
+ return *this = temp;
}
- balance_pair_t abs() const {
-#if 0
- balance_pair_t temp = *this;
- temp.quantity = temp.quantity.abs();
- if (temp.cost)
- temp.cost = temp.cost->abs();
- return temp;
-#else
- return balance_pair_t();
-#endif
+ virtual balance_t& in_place_unreduce() {
+ // A temporary must be used here because unreduction may cause
+ // multiple component amounts to collapse to the same commodity.
+ balance_t temp;
+ for (amounts_map::const_iterator i = amounts.begin();
+ i != amounts.end();
+ i++)
+ temp += i->second.unreduce();
+
+ if (cost) {
+ balance_t cost_temp;
+ for (amounts_map::const_iterator i = cost->amounts.begin();
+ i != cost->amounts.end();
+ i++)
+ cost_temp += i->second.unreduce();
+ return *this = balance_pair_t(temp, cost_temp);
+ }
+ return *this = temp;
}
- optional<amount_t>
- commodity_amount(const optional<const commodity_t&>& commodity = none) const {
- return quantity().commodity_amount(commodity);
+ balance_t& quantity() {
+ return *this;
}
- optional<balance_t> value(const optional<moment_t>& moment = none) const {
- return quantity().value(moment);
+ const balance_t& quantity() const {
+ return *this;
}
- balance_t
- strip_annotations(const bool keep_price = amount_t::keep_price,
- const bool keep_date = amount_t::keep_date,
- const bool keep_tag = amount_t::keep_tag) const {
- return quantity().strip_annotations(keep_price, keep_date, keep_tag);
- }
+ /**
+ * Truth tests. An balance pair may be truth tested by comparison
+ * to another balance pair, or by using one of the inherited
+ * operators from balance_t.
+ */
+ bool operator==(const balance_pair_t& bal_pair) const {
+ if (quantity() != bal_pair.quantity())
+ return false;
- void print(std::ostream& out, const int first_width,
- const int latter_width = -1) const {
- quantity().print(out, first_width, latter_width);
- }
+ if ((cost && ! bal_pair.cost) ||
+ (! cost && bal_pair.cost))
+ return false;
- balance_pair_t& add(const amount_t& amt,
- const optional<amount_t>& a_cost = none) {
-#if 0
- if (a_cost && ! cost)
- cost = quantity;
- quantity += amt;
- if (cost)
- *cost += a_cost ? *a_cost : amt;
-#endif
- return *this;
- }
+ if (*cost != *bal_pair.cost)
+ return false;
- bool valid() {
- return quantity().valid() && (! cost || cost->valid());
+ return true;
}
- void in_place_reduce() {
- quantity().in_place_reduce();
- if (cost) cost->in_place_reduce();
+ bool operator==(const balance_t& bal) const {
+ return balance_t::operator==(bal);
}
- balance_pair_t reduce() const {
- balance_pair_t temp(*this);
- temp.in_place_reduce();
- return temp;
+ bool operator==(const amount_t& amt) const {
+ return balance_t::operator==(amt);
+ }
+ template <typename T>
+ bool operator==(const T& val) const {
+ return balance_t::operator==(val);
}
- friend std::ostream& operator<<(std::ostream& out,
- const balance_pair_t& bal_pair);
-};
+ /**
+ * Debugging methods. There is only one method specifically for
+ * balance pairs to help with debugging:
+ *
+ * valid() returns true if the balances within the balance pair are
+ * valid.
+ */
+ virtual bool valid() {
+ if (! balance_t::valid())
+ return false;
+
+ if (cost && ! cost->valid())
+ return false;
-inline std::ostream& operator<<(std::ostream& out,
- const balance_pair_t& bal_pair) {
- bal_pair.quantity().print(out, 12);
- return out;
-}
+ return true;
+ }
+};
} // namespace ledger