diff --git a/example/CMakeLists.txt b/example/CMakeLists.txt index 0e7a71b0..76117aa4 100644 --- a/example/CMakeLists.txt +++ b/example/CMakeLists.txt @@ -22,6 +22,9 @@ cmake_minimum_required(VERSION 3.2) +add_library(example_utils INTERFACE) +target_include_directories(example_utils INTERFACE include) + # # add_example(target ...) # diff --git a/example/include/ranged_representation.h b/example/include/ranged_representation.h new file mode 100644 index 00000000..8d5ec234 --- /dev/null +++ b/example/include/ranged_representation.h @@ -0,0 +1,52 @@ +// 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. + +#pragma once + +#include "validated_type.h" +#include +#include + +template +inline constexpr auto is_in_range = [](const auto& v) { return std::clamp(v, Min, Max) == v; }; + +template +class ranged_representation : public validated_type)> { +public: + using base = validated_type)>; + using base::validated_type; + constexpr ranged_representation() : base(T{}) {} + + [[nodiscard]] constexpr ranged_representation operator-() const { return ranged_representation(-this->value()); } +}; + +template +struct std::common_type> : + std::type_identity< + ranged_representation, std::common_type_t{Min}, + std::common_type_t{Max}>> {}; + +template +struct std::common_type, std::intmax_t> : + std::type_identity< + ranged_representation, std::common_type_t{Min}, + std::common_type_t{Max}>> {}; diff --git a/example/include/validated_type.h b/example/include/validated_type.h new file mode 100644 index 00000000..bf44bbbc --- /dev/null +++ b/example/include/validated_type.h @@ -0,0 +1,81 @@ +// 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. + +#pragma once + +#include +#include +#include + +inline constexpr struct validated_tag { +} validated; + +template Validator> +class validated_type { + T value_; +public: + using value_type = T; + + static constexpr bool validate(const T& value) { return Validator()(value); } + + constexpr explicit validated_type(const T& value) noexcept(std::is_nothrow_copy_constructible_v) + requires std::copyable + : value_(value) + { + gsl_Expects(validate(value_)); + } + + constexpr explicit validated_type(T&& value) noexcept(std::is_nothrow_move_constructible_v) : + value_(std::move(value)) + { + gsl_Expects(validate(value_)); + } + + constexpr validated_type(const T& value, validated_tag) noexcept(std::is_nothrow_copy_constructible_v) + requires std::copyable + : value_(value) + { + } + + constexpr validated_type(T&& value, validated_tag) noexcept(std::is_nothrow_move_constructible_v) : + value_(std::move(value)) + { + } + + constexpr explicit(false) operator T() const noexcept(std::is_nothrow_copy_constructible_v) + requires std::copyable + { + return value_; + } + + constexpr T& value() & noexcept = delete; + constexpr const T& value() const& noexcept { return value_; } + constexpr T&& value() && noexcept { return std::move(value_); } + constexpr const T&& value() const&& noexcept { return std::move(value_); } + + bool operator==(const validated_type&) const + requires std::equality_comparable + = default; + auto operator<=>(const validated_type&) const + requires std::three_way_comparable + = default; +};