diff options
author | John Wiegley <johnw@newartisans.com> | 2019-03-15 17:24:46 -0700 |
---|---|---|
committer | John Wiegley <johnw@newartisans.com> | 2019-03-15 17:29:58 -0700 |
commit | 501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3 (patch) | |
tree | 9bfbdc9b2eb0c652d415526881002640a505b4cb | |
parent | d8d3183405999a2ed4c43a8fada76298ad089777 (diff) | |
download | fork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.tar.gz fork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.tar.bz2 fork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.zip |
Change compare_by_commodity to return an integer
-rw-r--r-- | src/balance.cc | 7 | ||||
-rw-r--r-- | src/commodity.cc | 177 | ||||
-rw-r--r-- | src/commodity.h | 2 | ||||
-rw-r--r-- | src/value.cc | 2 |
4 files changed, 108 insertions, 80 deletions
diff --git a/src/balance.cc b/src/balance.cc index 1ae9edb2..5f5e6fc9 100644 --- a/src/balance.cc +++ b/src/balance.cc @@ -244,8 +244,11 @@ void balance_t::sorted_amounts(amounts_array& sorted) const { foreach (const amounts_map::value_type& pair, amounts) sorted.push_back(&pair.second); - std::stable_sort(sorted.begin(), sorted.end(), - commodity_t::compare_by_commodity()); + std::stable_sort( + sorted.begin(), sorted.end(), + [](const amount_t * left, const amount_t * right) { + return commodity_t::compare_by_commodity()(left, right) < 0; + }); } void balance_t::map_sorted_amounts(function<void(const amount_t&)> fn) const diff --git a/src/commodity.cc b/src/commodity.cc index cf31886e..bdf6ee54 100644 --- a/src/commodity.cc +++ b/src/commodity.cc @@ -420,8 +420,8 @@ bool commodity_t::valid() const return true; } -bool commodity_t::compare_by_commodity::operator()(const amount_t * left, - const amount_t * right) const +int commodity_t::compare_by_commodity::operator()(const amount_t * left, + const amount_t * right) const { commodity_t& leftcomm(left->commodity()); commodity_t& rightcomm(right->commodity()); @@ -432,99 +432,124 @@ bool commodity_t::compare_by_commodity::operator()(const amount_t * left, int cmp = leftcomm.base_symbol().compare(rightcomm.base_symbol()); if (cmp != 0) { DEBUG("commodity.compare", "symbol is <"); - return cmp < 0; + return cmp; } - if (! leftcomm.has_annotation()) { - DEBUG("commodity.compare", "left has no annotation"); - return true; + if (! leftcomm.has_annotation() && rightcomm.has_annotation()) { + DEBUG("commodity.compare", "left has no annotation, right does"); + return -1; } - else if (! rightcomm.has_annotation()) { - DEBUG("commodity.compare", "right has no annotation"); - return false; + else if (leftcomm.has_annotation() && ! rightcomm.has_annotation()) { + DEBUG("commodity.compare", "right has no annotation, left does"); + return 1; + } + else if (! leftcomm.has_annotation() && ! rightcomm.has_annotation()) { + DEBUG("commodity.compare", "there are no annotations, commodities match"); + return 0; } - else { - annotated_commodity_t& aleftcomm(static_cast<annotated_commodity_t&>(leftcomm)); - annotated_commodity_t& arightcomm(static_cast<annotated_commodity_t&>(rightcomm)); - if (! aleftcomm.details.price && arightcomm.details.price) { - DEBUG("commodity.compare", "left has no price, right does"); - return true; - } - if (aleftcomm.details.price && ! arightcomm.details.price) { - DEBUG("commodity.compare", "right has no price, left does"); - return false; - } + annotated_commodity_t& aleftcomm(static_cast<annotated_commodity_t&>(leftcomm)); + annotated_commodity_t& arightcomm(static_cast<annotated_commodity_t&>(rightcomm)); - if (aleftcomm.details.price && arightcomm.details.price) { - amount_t leftprice(*aleftcomm.details.price); - amount_t rightprice(*arightcomm.details.price); + if (! aleftcomm.details.price && arightcomm.details.price) { + DEBUG("commodity.compare", "left has no price, right does"); + return -1; + } + if (aleftcomm.details.price && ! arightcomm.details.price) { + DEBUG("commodity.compare", "right has no price, left does"); + return 1; + } - if (leftprice.commodity() == rightprice.commodity()) { - DEBUG("commodity.compare", - "both have price, commodities match, comparing amount"); - return leftprice < rightprice; - } else { - // Since we have two different amounts, there's really no way - // to establish a true sorting order; we'll just do it based - // on the numerical values. - leftprice.clear_commodity(); - rightprice.clear_commodity(); - DEBUG("commodity.compare", - "both have price, commodities don't match, recursing"); - return commodity_t::compare_by_commodity()(&leftprice, &rightprice); + if (aleftcomm.details.price && arightcomm.details.price) { + amount_t leftprice(*aleftcomm.details.price); + amount_t rightprice(*arightcomm.details.price); + + if (leftprice.commodity() != rightprice.commodity()) { + // Since we have two different amounts, there's really no way + // to establish a true sorting order; we'll just do it based + // on the numerical values. + leftprice.clear_commodity(); + rightprice.clear_commodity(); + DEBUG("commodity.compare", + "both have price, commodities don't match, recursing"); + int cmp2 = commodity_t::compare_by_commodity()(&leftprice, &rightprice); + if (cmp2 != 0) { + DEBUG("commodity.compare", "recursion found a disparity"); + return cmp2; + } + } else { + if (leftprice < rightprice) { + DEBUG("commodity.compare", "left price is less"); + return -1; + } + else if (leftprice > rightprice) { + DEBUG("commodity.compare", "left price is more"); + return 1; } } + } - if (! aleftcomm.details.date && arightcomm.details.date) { - DEBUG("commodity.compare", "left has no date, right does"); - return true; - } - if (aleftcomm.details.date && ! arightcomm.details.date) { - DEBUG("commodity.compare", "right has no date, left does"); - return false; - } + if (! aleftcomm.details.date && arightcomm.details.date) { + DEBUG("commodity.compare", "left has no date, right does"); + return -1; + } + if (aleftcomm.details.date && ! arightcomm.details.date) { + DEBUG("commodity.compare", "right has no date, left does"); + return 1; + } - if (aleftcomm.details.date && arightcomm.details.date) { - gregorian::date_duration diff = - *aleftcomm.details.date - *arightcomm.details.date; - DEBUG("commodity.compare", "both have dates, comparing on difference"); - return diff.is_negative(); + if (aleftcomm.details.date && arightcomm.details.date) { + gregorian::date_duration diff = + *aleftcomm.details.date - *arightcomm.details.date; + DEBUG("commodity.compare", "both have dates, comparing on difference"); + if (diff.is_negative()) { + DEBUG("commodity.compare", "dates differ"); + return -1; } - if (! aleftcomm.details.tag && arightcomm.details.tag) { - DEBUG("commodity.compare", "left has no tag, right does"); - return true; - } - if (aleftcomm.details.tag && ! arightcomm.details.tag) { - DEBUG("commodity.compare", "right has no tag, left does"); - return false; + gregorian::date_duration diff2 = + *arightcomm.details.date - *aleftcomm.details.date; + if (diff2.is_negative()) { + DEBUG("commodity.compare", "dates differ"); + return 1; } + } - if (aleftcomm.details.tag && arightcomm.details.tag) { - DEBUG("commodity.compare", "both have tags, comparing lexically"); - return *aleftcomm.details.tag < *arightcomm.details.tag; - } + if (! aleftcomm.details.tag && arightcomm.details.tag) { + DEBUG("commodity.compare", "left has no tag, right does"); + return -1; + } + if (aleftcomm.details.tag && ! arightcomm.details.tag) { + DEBUG("commodity.compare", "right has no tag, left does"); + return 1; + } - if (! aleftcomm.details.value_expr && arightcomm.details.value_expr) { - DEBUG("commodity.compare", "left has no value expr, right does"); - return true; - } - if (aleftcomm.details.value_expr && ! arightcomm.details.value_expr) { - DEBUG("commodity.compare", "right has no value expr, left does"); - return false; - } + if (aleftcomm.details.tag && arightcomm.details.tag) { + DEBUG("commodity.compare", "both have tags, comparing lexically"); + if (*aleftcomm.details.tag < *arightcomm.details.tag) + return -1; + else if (*aleftcomm.details.tag > *arightcomm.details.tag) + return 1; + } - if (aleftcomm.details.value_expr && arightcomm.details.value_expr) { - DEBUG("commodity.compare", "both have value exprs, comparing text reprs"); - return (aleftcomm.details.value_expr->text() < - arightcomm.details.value_expr->text()); - } + if (! aleftcomm.details.value_expr && arightcomm.details.value_expr) { + DEBUG("commodity.compare", "left has no value expr, right does"); + return -1; + } + if (aleftcomm.details.value_expr && ! arightcomm.details.value_expr) { + DEBUG("commodity.compare", "right has no value expr, left does"); + return 1; + } - DEBUG("commodity.compare", "the two are incomparable, which should never happen"); - assert(false); - return true; + if (aleftcomm.details.value_expr && arightcomm.details.value_expr) { + DEBUG("commodity.compare", "both have value exprs, comparing text reprs"); + return (aleftcomm.details.value_expr->text() < + arightcomm.details.value_expr->text()); } + + DEBUG("commodity.compare", "the two are incomparable, which should never happen"); + assert(false); + return -1; } void put_commodity(property_tree::ptree& st, const commodity_t& comm, diff --git a/src/commodity.h b/src/commodity.h index c25ccaf2..5060102d 100644 --- a/src/commodity.h +++ b/src/commodity.h @@ -279,7 +279,7 @@ public: bool valid() const; struct compare_by_commodity { - bool operator()(const amount_t * left, const amount_t * right) const; + int operator()(const amount_t * left, const amount_t * right) const; }; }; diff --git a/src/value.cc b/src/value.cc index 5ce30530..18e78ab5 100644 --- a/src/value.cc +++ b/src/value.cc @@ -887,7 +887,7 @@ bool value_t::is_less_than(const value_t& val) const ! val.as_amount().has_commodity()) return as_amount() < val.as_amount(); else - return commodity_t::compare_by_commodity()(&as_amount(), &val.as_amount()); + return commodity_t::compare_by_commodity()(&as_amount(), &val.as_amount()) < 0; case BALANCE: return val.to_amount() > as_amount(); default: |