// The MIT License (MIT) // // Copyright (c) 2018 Mateusz Pusz // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. #include "units/si/velocity.h" #include #include using namespace units; using namespace units::literals; namespace { template class my_value { T data_{}; public: my_value() = default; constexpr my_value(T v) : data_{v} {} constexpr operator T() const { return data_; } }; } // namespace namespace units { template struct treat_as_floating_point> : std::is_floating_point { }; template struct quantity_values> { static constexpr my_value zero() { return my_value(0); } static constexpr my_value max() { return std::numeric_limits::max(); } static constexpr my_value min() { return std::numeric_limits::lowest(); } }; } // namespace units namespace std { template struct common_type, U> : common_type { }; template struct common_type> : common_type { }; } // namespace std namespace { using namespace units; using namespace units::literals; // class invariants // constexpr quantity> error(0_m); // should trigger a static_assert // constexpr quantity error(0); // should trigger a static_assert // constexpr quantity> error(0); // should trigger a static_assert // member types static_assert(std::is_same_v::rep, int>); static_assert(std::is_same_v::rep, float>); static_assert(std::is_same_v::ratio, std::ratio<1, 1>>); static_assert(std::is_same_v::ratio, std::ratio<1000, 1>>); // constructors static_assert(meters().count() == 0); constexpr meters kilometer{1000}; static_assert(kilometer.count() == 1000); static_assert(meters(kilometer).count() == kilometer.count()); static_assert(meters(1).count() == 1); static_assert(meters(my_value(1)).count() == 1); static_assert(meters>(1).count() == 1); // static_assert(meters(1.0).count() == 1); // should not compile // static_assert(meters(my_value(1.0)).count() == 1); // should not compile // static_assert(meters>(1.0).count() == 1); // should not compile static_assert(meters(1.0).count() == 1.0); static_assert(meters(my_value(1.0)).count() == 1.0); static_assert(meters(1).count() == 1.0); static_assert(meters(my_value(1)).count() == 1.0); static_assert(meters(3.14f).count() == 3.14f); static_assert(meters>(1.0).count() == 1.0); static_assert(meters>(1).count() == 1.0); static_assert(meters>(3.14f).count() == 3.14f); static_assert(meters(kilometer).count() == 1000); // static_assert(meters(meters(3.14)).count() == 3); // should not compile static_assert(meters(quantity_cast>(3.14_m)).count() == 3); // static_assert(meters(meters>(1000.0)).count() == 1000); // should not compile // static_assert(meters>(1000.0_m).count() == 1000); // should not compile static_assert(meters(1000.0_m).count() == 1000.0); static_assert(meters(meters>(1000.0)).count() == 1000.0); static_assert(meters>(1000.0_m).count() == 1000.0); static_assert(meters(kilometer).count() == 1000.0); static_assert(meters>(kilometer).count() == 1000.0); static_assert(meters(1_km).count() == 1000); // static_assert(meters(1_s).count() == 1); // should not compile // static_assert(kilometers(1010_m).count() == 1); // should not compile static_assert(kilometers(quantity_cast>(1010_m)).count() == 1); // assignment operator static_assert([]() { meters l1(1), l2(2); return l2 = l1; }() .count() == 1); // static member functions static_assert(meters::zero().count() == 0); static_assert(meters::min().count() == std::numeric_limits::lowest()); static_assert(meters::max().count() == std::numeric_limits::max()); static_assert(meters::zero().count() == 0.0); static_assert(meters::min().count() == std::numeric_limits::lowest()); static_assert(meters::max().count() == std::numeric_limits::max()); static_assert(meters>::zero().count() == 0); static_assert(meters>::min().count() == std::numeric_limits::lowest()); static_assert(meters>::max().count() == std::numeric_limits::max()); static_assert(meters>::zero().count() == 0.0); static_assert(meters>::min().count() == std::numeric_limits::lowest()); static_assert(meters>::max().count() == std::numeric_limits::max()); // unary member operators static_assert((+kilometer).count() == 1000); static_assert((-kilometer).count() == -1000); static_assert((+(-kilometer)).count() == -1000); static_assert((-(-kilometer)).count() == 1000); // binary member operators static_assert([](auto v) { auto vv = v++; return std::make_pair(v, vv); }(kilometer) == std::make_pair(meters(1001), meters(1000))); static_assert([](auto v) { auto vv = ++v; return std::make_pair(v, vv); }(kilometer) == std::make_pair(meters(1001), meters(1001))); static_assert([](auto v) { auto vv = v--; return std::make_pair(v, vv); }(kilometer) == std::make_pair(meters(999), meters(1000))); static_assert([](auto v) { auto vv = --v; return std::make_pair(v, vv); }(kilometer) == std::make_pair(meters(999), meters(999))); // compound assignment static_assert((1_m += 1_m).count() == 2); static_assert((2_m -= 1_m).count() == 1); static_assert((1_m *= 2).count() == 2); static_assert((2_m /= 2).count() == 1); static_assert((7_m %= 2).count() == 1); static_assert((7_m %= 2_m).count() == 1); // non-member arithmetic operators static_assert(std::is_same_v() + meters()), meters>); static_assert(std::is_same_v() - meters()), meters>); static_assert(std::is_same_v() * 1.0f), meters>); static_assert(std::is_same_v()), meters>); static_assert(std::is_same_v() / 1.0f), meters>); static_assert(std::is_same_v() / meters()), float>); static_assert(std::is_same_v() % short(1)), meters>); static_assert(std::is_same_v() % meters(1)), meters>); static_assert((1_m + kilometer).count() == 1001); static_assert((1_m + 1_km).count() == 1001); static_assert((kilometer - 1_m).count() == 999); static_assert((1_km - 1_m).count() == 999); static_assert((2_m * 2).count() == 4); static_assert((3 * 3_m).count() == 9); static_assert((4_m / 2).count() == 2); static_assert(4_m / 2_m == 2); static_assert(4_km / 2000_m == 2); static_assert((7_m % 2).count() == 1); static_assert((7_m % 2_m).count() == 1); static_assert((7_km % 2000_m).count() == 1000); // comparators static_assert(2_m + 1_m == 3_m); static_assert(!(2_m + 2_m == 3_m)); static_assert(2_m + 2_m != 3_m); static_assert(!(2_m + 2_m != 4_m)); static_assert(2_m > 1_m); static_assert(!(1_m > 1_m)); static_assert(1_m < 2_m); static_assert(!(2_m < 2_m)); static_assert(2_m >= 1_m); static_assert(2_m >= 2_m); static_assert(!(2_m >= 3_m)); static_assert(1_m <= 2_m); static_assert(2_m <= 2_m); static_assert(!(3_m <= 2_m)); static_assert(3_m == 3.0_m); static_assert(3_m != 3.14_m); static_assert(2_m > 1.0_m); static_assert(1.0_m < 2_m); static_assert(2.0_m >= 1_m); static_assert(1_m <= 2.0_m); static_assert(1000_m == 1_km); static_assert(1001_m != 1_km); static_assert(1001_m > 1_km); static_assert(999_m < 1_km); static_assert(1000_m >= 1_km); static_assert(1000_m <= 1_km); // is_quantity static_assert(units::detail::is_quantity>::value); // common_type static_assert(std::is_same_v, kilometers>, meters>); static_assert(std::is_same_v, meters>, meters>); static_assert(std::is_same_v, millimeters>, millimeters>); // quantity_cast // static_assert(quantity_cast(2_km).count() == 2000); // should not compile static_assert(quantity_cast>(2_km).count() == 2000); static_assert(quantity_cast>(2000_m).count() == 2); // time // static_assert(1_s == 1_m); // should not compile static_assert(1_h == 3600_s); // length static_assert(1_km == 1000_m); static_assert(1_km + 1_m == 1001_m); static_assert(10_km / 5_km == 2); static_assert(10_km / 2 == 5_km); // velocity static_assert(10_m / 5_s == 2_mps); static_assert(10 / 5_s * 1_m == 2_mps); static_assert(1_km / 1_s == 1000_mps); static_assert(2_kmph * 2_h == 4_km); static_assert(2_km / 2_kmph == 1_h); } // namespace