From 9c81cffc32a5e88ac9f81fe3d08c2a670efa79d2 Mon Sep 17 00:00:00 2001 From: Felix Morgner Date: Thu, 2 Jan 2020 12:48:57 +0100 Subject: new_type: implement multiplication --- include/newtype/impl/type_traits_extensions.hpp | 60 +++++++++++++++++++++++++ include/newtype/new_type.hpp | 18 ++++++++ 2 files changed, 78 insertions(+) (limited to 'include') diff --git a/include/newtype/impl/type_traits_extensions.hpp b/include/newtype/impl/type_traits_extensions.hpp index e9bbefb..c547d02 100644 --- a/include/newtype/impl/type_traits_extensions.hpp +++ b/include/newtype/impl/type_traits_extensions.hpp @@ -638,6 +638,66 @@ namespace nt::impl template auto constexpr is_nothrow_subtractable_v = is_nothrow_subtractable::value; + /** + * @brief A trait to test if a given type is multipliable + * + * @tparam T The type to test + * @note This specialization forms the base case for non-multipliable T + */ + template + struct is_multipliable : std::false_type + { + }; + + /** + * @brief A trait to test if a given type is multipliable + * + * @tparam T The type to test + * @note This specialization forms the case for multipliable T + */ + template + struct is_multipliable() * std::declval())>> : std::true_type + { + }; + + /** + * @brief A variable template to test if a given type is multipliable + * + * @tparam T The type to test + */ + template + auto constexpr is_multipliable_v = is_multipliable::value; + + /** + * @brief A trait to test if a given type is noexcept multipliable + * + * @tparam T The type to test + * @note This specialization forms the base case for non-noexcept multipliable or non-multipliable T + */ + template + struct is_nothrow_multipliable : std::false_type + { + }; + + /** + * @brief A trait to test if a given type is noexcept multipliable + * + * @tparam T The type to test + * @note This specialization forms the case for multipliable T detemining if T is noexcept multipliable + */ + template + struct is_nothrow_multipliable() * std::declval())>> + : std::bool_constant() * std::declval())> + { + }; + + /** + * @brief A variable template to test if a given type is noexcept multipliable + * + * @tparam T The type to test + */ + template + auto constexpr is_nothrow_multipliable_v = is_nothrow_multipliable::value; } // namespace arithmetic } // namespace nt::impl diff --git a/include/newtype/new_type.hpp b/include/newtype/new_type.hpp index bc8ecee..08d5024 100644 --- a/include/newtype/new_type.hpp +++ b/include/newtype/new_type.hpp @@ -364,6 +364,24 @@ namespace nt return {lhs.decay() - rhs.decay()}; } + /** + * @brief Multiply two instances of the same nt::new_type + * + * @note This operator is only available if the derivation clause of the passed in nt::new_type objects contains nt::Arithmetic and the base + * type is multipliable. + * @param lhs The left-hand side of the multiplication + * @param rhs The right-hand side of the multiplication + * @return a new instance of the same nt::new_type + */ + template + auto constexpr + operator*(new_type const & lhs, new_type const & rhs) noexcept( + impl::is_nothrow_multipliable_v && std::is_nothrow_copy_constructible_v) + -> std::enable_if_t, new_type> + { + return {lhs.decay() * rhs.decay()}; + } + } // namespace nt #endif -- cgit v1.2.3