summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2019-03-15 17:24:46 -0700
committerJohn Wiegley <johnw@newartisans.com>2019-03-15 17:29:58 -0700
commit501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3 (patch)
tree9bfbdc9b2eb0c652d415526881002640a505b4cb /src
parentd8d3183405999a2ed4c43a8fada76298ad089777 (diff)
downloadfork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.tar.gz
fork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.tar.bz2
fork-ledger-501fbc08ae5493db77bb34f4c4fbe1f3a3bc14e3.zip
Change compare_by_commodity to return an integer
Diffstat (limited to 'src')
-rw-r--r--src/balance.cc7
-rw-r--r--src/commodity.cc177
-rw-r--r--src/commodity.h2
-rw-r--r--src/value.cc2
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: