detected.hpp 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758
  1. #pragma once
  2. #include <type_traits>
  3. #include <nlohmann/detail/meta/void_t.hpp>
  4. // https://en.cppreference.com/w/cpp/experimental/is_detected
  5. namespace nlohmann
  6. {
  7. namespace detail
  8. {
  9. struct nonesuch
  10. {
  11. nonesuch() = delete;
  12. ~nonesuch() = delete;
  13. nonesuch(nonesuch const&) = delete;
  14. nonesuch(nonesuch const&&) = delete;
  15. void operator=(nonesuch const&) = delete;
  16. void operator=(nonesuch&&) = delete;
  17. };
  18. template <class Default,
  19. class AlwaysVoid,
  20. template <class...> class Op,
  21. class... Args>
  22. struct detector
  23. {
  24. using value_t = std::false_type;
  25. using type = Default;
  26. };
  27. template <class Default, template <class...> class Op, class... Args>
  28. struct detector<Default, void_t<Op<Args...>>, Op, Args...>
  29. {
  30. using value_t = std::true_type;
  31. using type = Op<Args...>;
  32. };
  33. template <template <class...> class Op, class... Args>
  34. using is_detected = typename detector<nonesuch, void, Op, Args...>::value_t;
  35. template <template <class...> class Op, class... Args>
  36. using detected_t = typename detector<nonesuch, void, Op, Args...>::type;
  37. template <class Default, template <class...> class Op, class... Args>
  38. using detected_or = detector<Default, void, Op, Args...>;
  39. template <class Default, template <class...> class Op, class... Args>
  40. using detected_or_t = typename detected_or<Default, Op, Args...>::type;
  41. template <class Expected, template <class...> class Op, class... Args>
  42. using is_detected_exact = std::is_same<Expected, detected_t<Op, Args...>>;
  43. template <class To, template <class...> class Op, class... Args>
  44. using is_detected_convertible =
  45. std::is_convertible<detected_t<Op, Args...>, To>;
  46. } // namespace detail
  47. } // namespace nlohmann