From 5d50c96317a08d8b395bc58722d50bb969fe0181 Mon Sep 17 00:00:00 2001 From: Marcel Greter Date: Mon, 19 Mar 2018 03:10:50 +0100 Subject: [PATCH] Cleanup AST values compare operators --- src/ast_values.cpp | 127 +++++++++++++++++++++------------------------ src/units.cpp | 4 ++ src/units.hpp | 1 + 3 files changed, 63 insertions(+), 69 deletions(-) diff --git a/src/ast_values.cpp b/src/ast_values.cpp index 78649b9b8..c0987cd65 100644 --- a/src/ast_values.cpp +++ b/src/ast_values.cpp @@ -84,15 +84,16 @@ namespace Sass { bool List::operator== (const Expression& rhs) const { - if (List_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { if (length() != r->length()) return false; if (separator() != r->separator()) return false; if (is_bracketed() != r->is_bracketed()) return false; for (size_t i = 0, L = length(); i < L; ++i) { - Expression_Obj rv = r->at(i); - Expression_Obj lv = this->at(i); - if (!lv || !rv) return false; - if (!(*lv == *rv)) return false; + auto rv = r->at(i); + auto lv = this->at(i); + if (!lv && rv) return false; + else if (!rv && lv) return false; + else if (*lv != *rv) return false; } return true; } @@ -141,13 +142,14 @@ namespace Sass { bool Map::operator== (const Expression& rhs) const { - if (Map_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { if (length() != r->length()) return false; for (auto key : keys()) { - Expression_Obj lv = at(key); - Expression_Obj rv = r->at(key); - if (!rv || !lv) return false; - if (!(*lv == *rv)) return false; + auto rv = r->at(key); + auto lv = this->at(key); + if (!lv && rv) return false; + else if (!rv && lv) return false; + else if (*lv != *rv) return false; } return true; } @@ -229,19 +231,12 @@ namespace Sass { bool Binary_Expression::operator==(const Expression& rhs) const { - try - { - Binary_Expression_Ptr_Const m = Cast(&rhs); - if (m == 0) return false; + if (auto m = Cast(&rhs)) { return type() == m->type() && - *left() == *m->left() && - *right() == *m->right(); + *left() == *m->left() && + *right() == *m->right(); } - catch (std::bad_cast&) - { - return false; - } - catch (...) { throw; } + return false; } size_t Binary_Expression::hash() const @@ -267,9 +262,9 @@ namespace Sass { bool Function::operator== (const Expression& rhs) const { - if (Function_Ptr_Const r = Cast(&rhs)) { - Definition_Ptr_Const d1 = Cast(definition()); - Definition_Ptr_Const d2 = Cast(r->definition()); + if (auto r = Cast(&rhs)) { + auto d1 = Cast(definition()); + auto d2 = Cast(r->definition()); return d1 && d2 && d1 == d2 && is_css() == r->is_css(); } return false; @@ -317,20 +312,14 @@ namespace Sass { bool Function_Call::operator==(const Expression& rhs) const { - try - { - Function_Call_Ptr_Const m = Cast(&rhs); - if (!(m && *sname() == *m->sname())) return false; - if (!(m && arguments()->length() == m->arguments()->length())) return false; - for (size_t i =0, L = arguments()->length(); i < L; ++i) - if (!(*(*arguments())[i] == *(*m->arguments())[i])) return false; + if (auto m = Cast(&rhs)) { + if (*sname() != *m->sname()) return false; + if (arguments()->length() != m->arguments()->length()) return false; + for (size_t i = 0, L = arguments()->length(); i < L; ++i) + if (*arguments()->get(i) != *m->arguments()->get(i)) return false; return true; } - catch (std::bad_cast&) - { - return false; - } - catch (...) { throw; } + return false; } size_t Function_Call::hash() const @@ -366,16 +355,10 @@ namespace Sass { bool Variable::operator==(const Expression& rhs) const { - try - { - Variable_Ptr_Const e = Cast(&rhs); - return e && name() == e->name(); - } - catch (std::bad_cast&) - { - return false; + if (auto e = Cast(&rhs)) { + return name() == e->name(); } - catch (...) { throw; } + return false; } size_t Variable::hash() @@ -452,21 +435,23 @@ namespace Sass { bool Number::operator== (const Expression& rhs) const { - if (auto rhsnr = Cast(&rhs)) { - return *this == *rhsnr; + if (auto n = Cast(&rhs)) { + return *this == *n; } return false; } bool Number::operator== (const Number& rhs) const { + // unitless or only having one unit are equivalent (3.4) + // therefore we need to reduce the units beforehand Number l(*this), r(rhs); l.reduce(); r.reduce(); size_t lhs_units = l.numerators.size() + l.denominators.size(); size_t rhs_units = r.numerators.size() + r.denominators.size(); - // unitless and only having one unit seems equivalent (will change in future) if (!lhs_units || !rhs_units) { return NEAR_EQUAL(l.value(), r.value()); } + // ensure both have same units l.normalize(); r.normalize(); Units &lhs_unit = l, &rhs_unit = r; return lhs_unit == rhs_unit && @@ -475,21 +460,26 @@ namespace Sass { bool Number::operator< (const Number& rhs) const { + // unitless or only having one unit are equivalent (3.4) + // therefore we need to reduce the units beforehand Number l(*this), r(rhs); l.reduce(); r.reduce(); size_t lhs_units = l.numerators.size() + l.denominators.size(); size_t rhs_units = r.numerators.size() + r.denominators.size(); - // unitless and only having one unit seems equivalent (will change in future) if (!lhs_units || !rhs_units) { return l.value() < r.value(); } + // ensure both have same units l.normalize(); r.normalize(); Units &lhs_unit = l, &rhs_unit = r; if (!(lhs_unit == rhs_unit)) { /* ToDo: do we always get usefull backtraces? */ throw Exception::IncompatibleUnits(rhs, *this); } - return lhs_unit < rhs_unit || - l.value() < r.value(); + if (lhs_unit == rhs_unit) { + return l.value() < r.value(); + } else { + return lhs_unit < rhs_unit; + } } ///////////////////////////////////////////////////////////////////////// @@ -512,7 +502,7 @@ namespace Sass { bool Color::operator== (const Expression& rhs) const { - if (Color_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { return r_ == r->r() && g_ == r->g() && b_ == r->b() && @@ -545,7 +535,7 @@ namespace Sass { bool Custom_Error::operator== (const Expression& rhs) const { - if (Custom_Error_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { return message() == r->message(); } return false; @@ -564,7 +554,7 @@ namespace Sass { bool Custom_Warning::operator== (const Expression& rhs) const { - if (Custom_Warning_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { return message() == r->message(); } return false; @@ -586,7 +576,7 @@ namespace Sass { bool Boolean::operator== (const Expression& rhs) const { - if (Boolean_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { return (value() == r->value()); } return false; @@ -642,13 +632,12 @@ namespace Sass { bool String_Schema::operator== (const Expression& rhs) const { - if (String_Schema_Ptr_Const r = Cast(&rhs)) { + if (auto r = Cast(&rhs)) { if (length() != r->length()) return false; for (size_t i = 0, L = length(); i < L; ++i) { - Expression_Obj rv = (*r)[i]; - Expression_Obj lv = (*this)[i]; - if (!lv || !rv) return false; - if (!(*lv == *rv)) return false; + auto rv = (*r)[i]; + auto lv = (*this)[i]; + if (*lv != *rv) return false; } return true; } @@ -707,10 +696,10 @@ namespace Sass { bool String_Constant::operator== (const Expression& rhs) const { - if (String_Quoted_Ptr_Const qstr = Cast(&rhs)) { - return (value() == qstr->value()); - } else if (String_Constant_Ptr_Const cstr = Cast(&rhs)) { - return (value() == cstr->value()); + if (auto qstr = Cast(&rhs)) { + return value() == qstr->value(); + } else if (auto cstr = Cast(&rhs)) { + return value() == cstr->value(); } return false; } @@ -753,10 +742,10 @@ namespace Sass { bool String_Quoted::operator== (const Expression& rhs) const { - if (String_Quoted_Ptr_Const qstr = Cast(&rhs)) { - return (value() == qstr->value()); - } else if (String_Constant_Ptr_Const cstr = Cast(&rhs)) { - return (value() == cstr->value()); + if (auto qstr = Cast(&rhs)) { + return value() == qstr->value(); + } else if (auto cstr = Cast(&rhs)) { + return value() == cstr->value(); } return false; } @@ -778,7 +767,7 @@ namespace Sass { bool Null::operator== (const Expression& rhs) const { - return rhs.concrete_type() == NULL_VAL; + return Cast(&rhs) != NULL; } size_t Null::hash() const diff --git a/src/units.cpp b/src/units.cpp index 779f1d2b4..ed99f5b2c 100644 --- a/src/units.cpp +++ b/src/units.cpp @@ -266,6 +266,10 @@ namespace Sass { return (numerators == rhs.numerators) && (denominators == rhs.denominators); } + bool Units::operator!= (const Units& rhs) const + { + return ! (*this == rhs); + } double Units::normalize() { diff --git a/src/units.hpp b/src/units.hpp index 306f5349b..3eb7369bc 100644 --- a/src/units.hpp +++ b/src/units.hpp @@ -83,6 +83,7 @@ namespace Sass { // compare operations bool operator< (const Units& rhs) const; bool operator== (const Units& rhs) const; + bool operator!= (const Units& rhs) const; // factor to convert into given units double convert_factor(const Units&) const; };