#ifndef _BALANCE_H #define _BALANCE_H #include "amount.h" #include #include #include namespace ledger { typedef std::map amounts_map; typedef std::pair amounts_pair; class balance_t { public: amounts_map amounts; bool valid() const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if (! (*i).second.valid()) return false; return true; } // constructors balance_t() {} balance_t(const balance_t& bal) { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) *this += (*i).second; } balance_t(const amount_t& amt) { if (amt) amounts.insert(amounts_pair(&amt.commodity(), amt)); } template balance_t(T value) { amount_t amt(value); if (amt) amounts.insert(amounts_pair(&amt.commodity(), amt)); } // assignment operator balance_t& operator=(const balance_t& bal) { if (this != &bal) { amounts.clear(); for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) *this += (*i).second; } return *this; } balance_t& operator=(const amount_t& amt) { amounts.clear(); *this += amt; return *this; } template balance_t& operator=(T value) { amounts.clear(); *this += value; return *this; } // in-place arithmetic balance_t& operator+=(const balance_t& bal) { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) *this += (*i).second; return *this; } balance_t& operator+=(const amount_t& amt) { amounts_map::iterator i = amounts.find(&amt.commodity()); if (i != amounts.end()) (*i).second += amt; else if (amt) amounts.insert(amounts_pair(&amt.commodity(), amt)); return *this; } template balance_t& operator+=(T val) { return *this += amount_t(val); } balance_t& operator-=(const balance_t& bal) { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) *this -= (*i).second; return *this; } balance_t& operator-=(const amount_t& amt) { amounts_map::iterator i = amounts.find(&amt.commodity()); if (i != amounts.end()) (*i).second -= amt; else if (amt) amounts.insert(amounts_pair(&amt.commodity(), amt)); return *this; } template balance_t& operator-=(T val) { return *this -= amount_t(val); } // simple arithmetic balance_t operator+(const balance_t& bal) const { balance_t temp = *this; temp += bal; return temp; } balance_t operator+(const amount_t& amt) const { balance_t temp = *this; temp += amt; return temp; } template balance_t operator+(T val) const { balance_t temp = *this; temp += val; return temp; } balance_t operator-(const balance_t& bal) const { balance_t temp = *this; temp -= bal; return temp; } balance_t operator-(const amount_t& amt) const { balance_t temp = *this; temp -= amt; return temp; } template balance_t operator-(T val) const { balance_t temp = *this; temp -= val; return temp; } // multiplication and divide balance_t& operator*=(const balance_t& bal) { if (amounts.size() == 1 && bal.amounts.size() == 1) return *this *= (*bal.amounts.begin()).second; throw amount_error("It makes no sense to multiply two balances"); } balance_t& operator*=(const amount_t& amt) { // Multiplying by the null commodity causes all amounts to be // increased by the same factor. if (amt.commodity().symbol.empty()) { for (amounts_map::iterator i = amounts.begin(); i != amounts.end(); i++) (*i).second *= amt; } else if (amounts.size() == 1) { (*amounts.begin()).second *= amt; } else { amounts_map::iterator i = amounts.find(&amt.commodity()); if (i != amounts.end()) (*i).second *= amt; } return *this; } template balance_t& operator*=(T val) { return *this *= amount_t(val); } balance_t& operator/=(const balance_t& bal) { if (amounts.size() == 1 && bal.amounts.size() == 1) return *this /= (*bal.amounts.begin()).second; throw amount_error("It makes no sense to divide two balances"); } balance_t& operator/=(const amount_t& amt) { // Dividing by the null commodity causes all amounts to be // increased by the same factor. if (amt.commodity().symbol.empty()) { for (amounts_map::iterator i = amounts.begin(); i != amounts.end(); i++) (*i).second /= amt; } else if (amounts.size() == 1) { (*amounts.begin()).second /= amt; } else { amounts_map::iterator i = amounts.find(&amt.commodity()); if (i != amounts.end()) (*i).second /= amt; } return *this; } template balance_t& operator/=(T val) { return *this /= amount_t(val); } // multiplication and divide balance_t operator*(const balance_t& bal) const { balance_t temp = *this; temp *= bal; return temp; } balance_t operator*(const amount_t& amt) const { balance_t temp = *this; temp *= amt; return temp; } template balance_t operator*(T val) const { balance_t temp = *this; temp *= val; return temp; } balance_t operator/(const balance_t& bal) const { balance_t temp = *this; temp /= bal; return temp; } balance_t operator/(const amount_t& amt) const { balance_t temp = *this; temp /= amt; return temp; } template balance_t operator/(T val) const { balance_t temp = *this; temp /= val; return temp; } // comparison bool operator<(const balance_t& bal) const { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) if (! (amount(*(*i).first) < (*i).second)) return false; for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if (! ((*i).second < bal.amount(*(*i).first))) return false; if (bal.amounts.size() == 0 && amounts.size() == 0) return false; return true; } bool operator<(const amount_t& amt) const { return amount(amt.commodity()) < amt; } template bool operator<(T val) const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second < val) return true; return false; } bool operator<=(const balance_t& bal) const { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) if (! (amount(*(*i).first) <= (*i).second)) return false; for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if (! ((*i).second <= bal.amount(*(*i).first))) return false; return true; } bool operator<=(const amount_t& amt) const { return amount(amt.commodity()) <= amt; } template bool operator<=(T val) const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second <= val) return true; return false; } bool operator>(const balance_t& bal) const { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) if (! (amount(*(*i).first) > (*i).second)) return false; for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if (! ((*i).second > bal.amount(*(*i).first))) return false; if (bal.amounts.size() == 0 && amounts.size() == 0) return false; return true; } bool operator>(const amount_t& amt) const { return amount(amt.commodity()) > amt; } template bool operator>(T val) const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second > val) return true; return false; } bool operator>=(const balance_t& bal) const { for (amounts_map::const_iterator i = bal.amounts.begin(); i != bal.amounts.end(); i++) if (! (amount(*(*i).first) >= (*i).second)) return false; for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if (! ((*i).second >= bal.amount(*(*i).first))) return false; return true; } bool operator>=(const amount_t& amt) const { return amount(amt.commodity()) >= amt; } template bool operator>=(T val) const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second >= val) return true; return false; } bool operator==(const balance_t& bal) const { amounts_map::const_iterator i, j; for (i = amounts.begin(), j = bal.amounts.begin(); i != amounts.end() && j != bal.amounts.end(); i++, j++) { if (! ((*i).first == (*j).first && (*i).second == (*j).second)) return false; } return i == amounts.end() && j == bal.amounts.end(); } bool operator==(const amount_t& amt) const { return amounts.size() == 1 && (*amounts.begin()).second == amt; } template bool operator==(T val) const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second == val) return true; return false; } bool operator!=(const balance_t& bal) const { return ! (*this == bal); } bool operator!=(const amount_t& amt) const { return ! (*this == amt); } template bool operator!=(T val) const { return ! (*this == val); } // unary negation void negate() { for (amounts_map::iterator i = amounts.begin(); i != amounts.end(); i++) (*i).second.negate(); } balance_t negated() const { balance_t temp = *this; temp.negate(); return temp; } balance_t operator-() const { return negated(); } // conversion operators operator amount_t() const { if (amounts.size() == 1) return (*amounts.begin()).second; else if (amounts.size() == 0) return amount_t(); else throw amount_error("Cannot convert a balance with " "multiple commodities to an amount"); } operator bool() const { for (amounts_map::const_iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second) return true; return false; } amount_t amount(const commodity_t& commodity) const; balance_t value(const std::time_t moment) const; balance_t price() const; void write(std::ostream& out, const int first_width, const int latter_width = -1) const; void abs() { for (amounts_map::iterator i = amounts.begin(); i != amounts.end(); i++) (*i).second.abs(); } void round() { for (amounts_map::iterator i = amounts.begin(); i != amounts.end(); i++) if ((*i).second.commodity()) (*i).second = (*i).second.round((*i).second.commodity().precision()); } }; inline balance_t abs(const balance_t& bal) { balance_t temp = bal; temp.abs(); return temp; } inline std::ostream& operator<<(std::ostream& out, const balance_t& bal) { bal.write(out, 12); return out; } class balance_pair_t { public: balance_t quantity; balance_t * price; balance_t * cost; // constructors balance_pair_t() : price(NULL), cost(NULL) {} balance_pair_t(const balance_pair_t& bal_pair) : quantity(bal_pair.quantity), price(NULL), cost(NULL) { if (bal_pair.price) price = new balance_t(*bal_pair.price); if (bal_pair.cost) cost = new balance_t(*bal_pair.cost); } balance_pair_t(const balance_t& _quantity) : quantity(_quantity), price(NULL), cost(NULL) {} balance_pair_t(const amount_t& _quantity) : quantity(_quantity), price(NULL), cost(NULL) {} template balance_pair_t(T value) : quantity(value), price(NULL), cost(NULL) {} // destructor ~balance_pair_t() { if (price) delete price; if (cost) delete cost; } // assignment operator balance_pair_t& operator=(const balance_pair_t& bal_pair) { if (this != &bal_pair) { if (price) { delete price; price = NULL; } if (cost) { delete cost; cost = NULL; } quantity = bal_pair.quantity; if (bal_pair.price) price = new balance_t(*bal_pair.price); if (bal_pair.cost) cost = new balance_t(*bal_pair.cost); } return *this; } balance_pair_t& operator=(const balance_t& bal) { if (price) { delete price; price = NULL; } if (cost) { delete cost; cost = NULL; } quantity = bal; return *this; } balance_pair_t& operator=(const amount_t& amt) { if (price) { delete price; price = NULL; } if (cost) { delete cost; cost = NULL; } quantity = amt; return *this; } template balance_pair_t& operator=(T value) { if (price) { delete price; price = NULL; } if (cost) { delete cost; cost = NULL; } quantity = value; return *this; } // in-place arithmetic balance_pair_t& operator+=(const balance_pair_t& bal_pair) { quantity += bal_pair.quantity; if (bal_pair.price) { if (price) *price += *bal_pair.price; else price = new balance_t(*bal_pair.price); } if (bal_pair.cost) { if (cost) *cost += *bal_pair.cost; else cost = new balance_t(*bal_pair.cost); } return *this; } balance_pair_t& operator+=(const balance_t& bal) { quantity += bal; return *this; } balance_pair_t& operator+=(const amount_t& amt) { quantity += amt; return *this; } template balance_pair_t& operator+=(T val) { return *this += amount_t(val); } balance_pair_t& operator-=(const balance_pair_t& bal_pair) { quantity -= bal_pair.quantity; if (bal_pair.price) { if (price) *price -= *bal_pair.price; else price = new balance_t(- *bal_pair.price); } if (bal_pair.cost) { if (cost) *cost -= *bal_pair.cost; else cost = new balance_t(- *bal_pair.cost); } return *this; } balance_pair_t& operator-=(const balance_t& bal) { quantity -= bal; return *this; } balance_pair_t& operator-=(const amount_t& amt) { quantity -= amt; return *this; } template balance_pair_t& operator-=(T val) { return *this -= amount_t(val); } // simple arithmetic balance_pair_t operator+(const balance_pair_t& bal_pair) const { balance_pair_t temp = *this; temp += bal_pair; return temp; } balance_pair_t operator+(const balance_t& bal) const { balance_pair_t temp = *this; temp += bal; return temp; } balance_pair_t operator+(const amount_t& amt) const { balance_pair_t temp = *this; temp += amt; return temp; } template balance_pair_t operator+(T val) const { balance_pair_t temp = *this; temp += val; return temp; } balance_pair_t operator-(const balance_pair_t& bal_pair) const { balance_pair_t temp = *this; temp -= bal_pair; return temp; } balance_pair_t operator-(const balance_t& bal) const { balance_pair_t temp = *this; temp -= bal; return temp; } balance_pair_t operator-(const amount_t& amt) const { balance_pair_t temp = *this; temp -= amt; return temp; } template balance_pair_t operator-(T val) const { balance_pair_t temp = *this; temp -= val; return temp; } // multiplication and division balance_pair_t& operator*=(const balance_pair_t& bal_pair) { quantity *= bal_pair.quantity; if (bal_pair.price) { if (price) *price *= *bal_pair.price; } else { if (price) { delete price; price = NULL; } } if (bal_pair.cost) { if (cost) *cost *= *bal_pair.cost; } else { if (cost) { delete cost; cost = NULL; } } return *this; } balance_pair_t& operator*=(const balance_t& bal) { quantity *= bal; return *this; } balance_pair_t& operator*=(const amount_t& amt) { quantity *= amt; return *this; } template balance_pair_t& operator*=(T val) { return *this *= amount_t(val); } balance_pair_t& operator/=(const balance_pair_t& bal_pair) { quantity /= bal_pair.quantity; if (bal_pair.price) { if (price) *price /= *bal_pair.price; } else { throw amount_error("Attempt to divide by zero"); } if (bal_pair.cost) { if (cost) *cost /= *bal_pair.cost; } else { throw amount_error("Attempt to divide by zero"); } return *this; } balance_pair_t& operator/=(const balance_t& bal) { quantity /= bal; return *this; } balance_pair_t& operator/=(const amount_t& amt) { quantity /= amt; return *this; } template balance_pair_t& operator/=(T val) { return *this /= amount_t(val); } balance_pair_t operator*(const balance_pair_t& bal_pair) const { balance_pair_t temp = *this; temp *= bal_pair; return temp; } balance_pair_t operator*(const balance_t& bal) const { balance_pair_t temp = *this; temp *= bal; return temp; } balance_pair_t operator*(const amount_t& amt) const { balance_pair_t temp = *this; temp *= amt; return temp; } template balance_pair_t operator*(T val) const { balance_pair_t temp = *this; temp *= val; return temp; } balance_pair_t operator/(const balance_pair_t& bal_pair) const { balance_pair_t temp = *this; temp /= bal_pair; return temp; } balance_pair_t operator/(const balance_t& bal) const { balance_pair_t temp = *this; temp /= bal; return temp; } balance_pair_t operator/(const amount_t& amt) const { balance_pair_t temp = *this; temp /= amt; return temp; } template balance_pair_t operator/(T val) const { balance_pair_t temp = *this; temp /= val; return temp; } // 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; } template bool operator<(T val) const { return quantity < val; } 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; } template bool operator<=(T val) const { return quantity <= val; } 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; } template bool operator>(T val) const { return quantity > val; } 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; } template bool operator>=(T val) const { return quantity >= val; } 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; } template bool operator==(T val) const { return quantity == val; } bool operator!=(const balance_pair_t& bal_pair) const { return ! (*this == bal_pair); } bool operator!=(const balance_t& bal) const { return ! (*this == bal); } bool operator!=(const amount_t& amt) const { return ! (*this == amt); } template bool operator!=(T val) const { return ! (*this == val); } // unary negation void negate() { quantity.negate(); if (price) price->negate(); if (cost) cost->negate(); } balance_pair_t negated() const { balance_pair_t temp = *this; temp.negate(); return temp; } balance_pair_t operator-() const { return negated(); } // test for non-zero (use ! for zero) operator bool() const { return quantity; } operator balance_t() const { return quantity; } operator amount_t() const { return quantity; } void abs() { quantity.abs(); if (price) price->abs(); if (cost) cost->abs(); } amount_t amount(const commodity_t& commodity) const { return quantity.amount(commodity); } balance_t value(const std::time_t moment) const { return quantity.value(moment); } void write(std::ostream& out, const int first_width, const int latter_width = -1) const { quantity.write(out, first_width, latter_width); } balance_pair_t& add(const amount_t& amount, const amount_t * a_price = NULL, const amount_t * a_cost = NULL) { quantity += amount; if (a_price) { if (price) *price += *a_price; else price = new balance_t(*a_price); } if (a_cost) { if (cost) *cost += *a_cost; else cost = new balance_t(*a_cost); } return *this; } bool valid() { return (quantity.valid() && (! price || price->valid()) && (! cost || cost->valid())); } void round() { quantity.round(); } }; inline balance_pair_t abs(const balance_pair_t& bal_pair) { balance_pair_t temp; temp.abs(); return temp; } inline std::ostream& operator<<(std::ostream& out, const balance_pair_t& bal_pair) { bal_pair.quantity.write(out, 12); return out; } } // namespace ledger #endif // _BALANCE_H