// __ __ _ ______ _____ // | \/ | (_) | ____| / ____|_ _ // | \ / | __ _ __ _ _ ___ | |__ _ __ _ _ _ __ ___ | | _| |_ _| |_ // | |\/| |/ _` |/ _` | |/ __| | __| | '_ \| | | | '_ ` _ \ | | |_ _|_ _| // | | | | (_| | (_| | | (__ | |____| | | | |_| | | | | | | | |____|_| |_| // |_| |_|\__,_|\__, |_|\___| |______|_| |_|\__,_|_| |_| |_| \_____| // __/ | https://github.com/Neargye/magic_enum // |___/ version 0.8.2 // // Licensed under the MIT License . // SPDX-License-Identifier: MIT // Copyright (c) 2019 - 2022 Daniil Goncharov . // // 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. #ifndef NEARGYE_MAGIC_ENUM_SWITCH_HPP #define NEARGYE_MAGIC_ENUM_SWITCH_HPP #include "magic_enum.hpp" namespace magic_enum { namespace detail { struct default_result_type {}; template struct identity { using type = T; }; struct nonesuch {}; template > struct invoke_result : identity {}; template struct invoke_result : std::invoke_result {}; template using invoke_result_t = typename invoke_result::type; template constexpr auto common_invocable(std::index_sequence) noexcept { static_assert(is_enum_v, "magic_enum::detail::invocable_index requires enum type."); if constexpr (count_v == 0) { return identity{}; } else { return std::common_type[I]>>...>{}; } } template constexpr auto result_type() noexcept { static_assert(is_enum_v, "magic_enum::detail::result_type requires enum type."); constexpr auto seq = std::make_index_sequence>{}; using R = typename decltype(common_invocable(seq))::type; if constexpr (std::is_same_v) { if constexpr (std::is_same_v) { return identity{}; } else { return identity{}; } } else { if constexpr (std::is_convertible_v) { return identity{}; } else if constexpr (std::is_convertible_v) { return identity{}; } else { return identity{}; } } } template , typename R = typename decltype(result_type())::type> using result_t = std::enable_if_t && !std::is_same_v, R>; #if !defined(MAGIC_ENUM_ENABLE_HASH) template inline constexpr auto default_result_type_lambda = []() noexcept(std::is_nothrow_default_constructible_v) { return T{}; }; template <> inline constexpr auto default_result_type_lambda = []() noexcept {}; template constexpr R invoke_r(F&& f, Args&&... args) noexcept(std::is_nothrow_invocable_r_v) { if constexpr (std::is_void_v) { std::forward(f)(std::forward(args)...); } else { return static_cast(std::forward(f)(std::forward(args)...)); } } template constexpr decltype(auto) constexpr_switch_impl(F&& f, E value, Def&& def) { if constexpr(I < End) { constexpr auto v = enum_constant()>{}; if (value == v) { if constexpr (std::is_invocable_r_v) { return invoke_r(std::forward(f), v); } else { return def(); } } else { return constexpr_switch_impl(std::forward(f), value, std::forward(def)); } } else { return def(); } } template constexpr decltype(auto) constexpr_switch(F&& f, E value, Def&& def) { static_assert(is_enum_v, "magic_enum::detail::constexpr_switch requires enum type."); if constexpr (count_v == 0) { return def(); } else { return constexpr_switch_impl<0, count_v, R>(std::forward(f), value, std::forward(def)); } } #endif } // namespace magic_enum::detail template > constexpr decltype(auto) enum_switch(F&& f, E value) { using D = std::decay_t; static_assert(std::is_enum_v, "magic_enum::enum_switch requires enum type."); #if defined(MAGIC_ENUM_ENABLE_HASH) return detail::constexpr_switch<&detail::values_v, detail::case_call_t::value>( std::forward(f), value, detail::default_result_type_lambda); #else return detail::constexpr_switch( std::forward(f), value, detail::default_result_type_lambda); #endif } template > constexpr decltype(auto) enum_switch(F&& f, E value, Result&& result) { using D = std::decay_t; static_assert(std::is_enum_v, "magic_enum::enum_switch requires enum type."); #if defined(MAGIC_ENUM_ENABLE_HASH) return detail::constexpr_switch<&detail::values_v, detail::case_call_t::value>( std::forward(f), value, [&result]() -> R { return std::forward(result); }); #else return detail::constexpr_switch( std::forward(f), value, [&result]() -> R { return std::forward(result); }); #endif } } // namespace magic_enum template <> struct std::common_type : magic_enum::detail::identity {}; template struct std::common_type : magic_enum::detail::identity {}; template struct std::common_type : magic_enum::detail::identity {}; #endif // NEARGYE_MAGIC_ENUM_SWITCH_HPP