mirror of
https://github.com/BlackMATov/meta.hpp.git
synced 2025-12-13 11:17:06 +07:00
339 lines
11 KiB
C++
339 lines
11 KiB
C++
/*******************************************************************************
|
|
* This file is part of the "https://github.com/blackmatov/meta.hpp"
|
|
* For conditions of distribution and use, see copyright notice in LICENSE.md
|
|
* Copyright (C) 2021-2024, by Matvey Cherevko (blackmatov@gmail.com)
|
|
******************************************************************************/
|
|
|
|
#include <meta.hpp/meta_all.hpp>
|
|
#include <doctest/doctest.h>
|
|
|
|
namespace
|
|
{
|
|
struct base0 {
|
|
int i{21};
|
|
};
|
|
|
|
struct base1 : virtual base0 {
|
|
int j{42};
|
|
META_HPP_ENABLE_BASE_INFO(base0)
|
|
};
|
|
|
|
struct base2 : virtual base0 {
|
|
int k{84};
|
|
META_HPP_ENABLE_BASE_INFO(base0)
|
|
};
|
|
|
|
struct derived : base1, base2 {
|
|
int l{168};
|
|
META_HPP_ENABLE_BASE_INFO(base1, base2)
|
|
};
|
|
|
|
struct derived2 : base1, base2 {
|
|
int m{336};
|
|
META_HPP_ENABLE_BASE_INFO(base1, base2)
|
|
};
|
|
}
|
|
|
|
TEST_CASE("meta/meta_utilities/value3/get_type") {
|
|
namespace meta = meta_hpp;
|
|
|
|
SUBCASE("from ref") {
|
|
{
|
|
derived d{};
|
|
CHECK(meta::uvalue{d}.get_type() == meta::resolve_type<derived>());
|
|
}
|
|
{
|
|
const derived d{};
|
|
CHECK(meta::uvalue{d}.get_type() == meta::resolve_type<derived>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("from ptr") {
|
|
{
|
|
derived d{};
|
|
derived* pd = &d;
|
|
CHECK(meta::uvalue{pd}.get_type() == meta::resolve_type<derived*>());
|
|
}
|
|
{
|
|
derived d{};
|
|
derived* const pd = &d;
|
|
CHECK(meta::uvalue{pd}.get_type() == meta::resolve_type<derived*>());
|
|
}
|
|
{
|
|
derived d{};
|
|
const derived* pd = &d;
|
|
CHECK(meta::uvalue{pd}.get_type() == meta::resolve_type<const derived*>());
|
|
}
|
|
{
|
|
derived d{};
|
|
const derived* const pd = &d;
|
|
CHECK(meta::uvalue{pd}.get_type() == meta::resolve_type<const derived*>());
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_CASE("meta/meta_utilities/value3/is") {
|
|
namespace meta = meta_hpp;
|
|
|
|
{
|
|
CHECK(meta::uvalue{derived{}}.is<derived>());
|
|
CHECK_FALSE(meta::uvalue{derived{}}.is<derived2>());
|
|
}
|
|
|
|
{
|
|
CHECK(meta::uvalue{derived{}}.is<base1>());
|
|
CHECK(meta::uvalue{derived{}}.is<base2>());
|
|
}
|
|
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.is<derived*>());
|
|
CHECK_FALSE(v.is<const derived*>());
|
|
CHECK_FALSE(v.is<void*>());
|
|
CHECK_FALSE(v.is<const void*>());
|
|
}
|
|
|
|
{
|
|
meta::uvalue v{nullptr};
|
|
CHECK(v.is<std::nullptr_t>());
|
|
CHECK_FALSE(v.is<void*>());
|
|
}
|
|
}
|
|
|
|
TEST_CASE("meta/meta_utilities/value3/as") {
|
|
namespace meta = meta_hpp;
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().as<derived>()), derived&>);
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&&>().as<derived>()), derived>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().as<derived>()), const derived&>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&&>().as<derived>()), const derived&&>);
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&&>().as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&&>().as<derived*>()), derived*>);
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&&>().as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&&>().as<const derived*>()), const derived*>);
|
|
|
|
SUBCASE("derived to derived") {
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK(v.as<derived>().l == 168);
|
|
CHECK_THROWS(std::ignore = v.as<derived2>());
|
|
}
|
|
{
|
|
const meta::uvalue v{derived{}};
|
|
CHECK(v.as<derived>().l == 168);
|
|
CHECK_THROWS(std::ignore = v.as<derived2>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived to base") {
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK(v.as<base2>().k == 84);
|
|
}
|
|
{
|
|
const meta::uvalue v{derived{}};
|
|
CHECK(v.as<base2>().k == 84);
|
|
}
|
|
}
|
|
|
|
SUBCASE("voidptr") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.as<void*>() == &d);
|
|
CHECK(v.as<const void*>() == &d);
|
|
}
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK_THROWS(std::ignore = v.as<void*>());
|
|
CHECK(v.as<const void*>() == &d);
|
|
}
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK_THROWS(std::ignore = v.as<void*>());
|
|
CHECK_THROWS(std::ignore = v.as<const void*>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("nullptr") {
|
|
{
|
|
meta::uvalue v{nullptr};
|
|
CHECK(v.as<void*>() == nullptr);
|
|
CHECK(v.as<const void*>() == nullptr);
|
|
CHECK(v.as<derived*>() == nullptr);
|
|
CHECK(v.as<const derived*>() == nullptr);
|
|
|
|
CHECK_THROWS(std::ignore = v.as<derived>());
|
|
}
|
|
{
|
|
const meta::uvalue v{nullptr};
|
|
CHECK(v.as<void*>() == nullptr);
|
|
CHECK(v.as<const void*>() == nullptr);
|
|
CHECK(v.as<derived*>() == nullptr);
|
|
CHECK(v.as<const derived*>() == nullptr);
|
|
|
|
CHECK_THROWS(std::ignore = v.as<derived>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived* to derived*") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.as<derived*>()->l == 168);
|
|
CHECK(v.as<const derived*>()->l == 168);
|
|
|
|
CHECK_THROWS(std::ignore = v.as<derived2*>());
|
|
CHECK_THROWS(std::ignore = v.as<const derived2*>());
|
|
}
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.as<const derived*>()->l == 168);
|
|
|
|
CHECK_THROWS(std::ignore = v.as<const derived2*>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived* to base*") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.as<base2*>()->k == 84);
|
|
CHECK(v.as<const base2*>()->k == 84);
|
|
}
|
|
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK_THROWS(std::ignore = v.as<base2*>());
|
|
CHECK(v.as<const base2*>()->k == 84);
|
|
}
|
|
}
|
|
}
|
|
|
|
TEST_CASE("meta/meta_utilities/value3/try_as") {
|
|
namespace meta = meta_hpp;
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().try_as<derived>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().try_as<derived>()), const derived*>);
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().try_as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&&>().try_as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().try_as<derived*>()), derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&&>().try_as<derived*>()), derived*>);
|
|
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&>().try_as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<meta::uvalue&&>().try_as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&>().try_as<const derived*>()), const derived*>);
|
|
static_assert(std::is_same_v<decltype(std::declval<const meta::uvalue&&>().try_as<const derived*>()), const derived*>);
|
|
|
|
SUBCASE("derived to derived") {
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK(v.try_as<derived>()->l == 168);
|
|
CHECK_FALSE(v.try_as<derived2>());
|
|
}
|
|
{
|
|
const meta::uvalue v{derived{}};
|
|
CHECK(v.try_as<derived>()->l == 168);
|
|
CHECK_FALSE(v.try_as<derived2>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived to base") {
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK(v.try_as<base2>()->k == 84);
|
|
}
|
|
{
|
|
const meta::uvalue v{derived{}};
|
|
CHECK(v.try_as<base2>()->k == 84);
|
|
}
|
|
}
|
|
|
|
SUBCASE("voidptr") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.try_as<void*>() == &d);
|
|
CHECK(v.try_as<const void*>() == &d);
|
|
}
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK_FALSE(v.try_as<void*>());
|
|
CHECK(v.try_as<const void*>() == &d);
|
|
}
|
|
{
|
|
meta::uvalue v{derived{}};
|
|
CHECK_FALSE(v.try_as<void*>());
|
|
CHECK_FALSE(v.try_as<const void*>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("nullptr") {
|
|
{
|
|
meta::uvalue v{nullptr};
|
|
CHECK(v.try_as<void*>() == nullptr);
|
|
CHECK(v.try_as<const void*>() == nullptr);
|
|
CHECK(v.try_as<derived*>() == nullptr);
|
|
CHECK(v.try_as<const derived*>() == nullptr);
|
|
|
|
CHECK_FALSE(v.try_as<derived>());
|
|
}
|
|
{
|
|
const meta::uvalue v{nullptr};
|
|
CHECK(v.try_as<void*>() == nullptr);
|
|
CHECK(v.try_as<const void*>() == nullptr);
|
|
CHECK(v.try_as<derived*>() == nullptr);
|
|
CHECK(v.try_as<const derived*>() == nullptr);
|
|
|
|
CHECK_FALSE(v.try_as<derived>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived* to derived*") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.try_as<derived*>()->l == 168);
|
|
CHECK(v.try_as<const derived*>()->l == 168);
|
|
|
|
CHECK_FALSE(v.try_as<derived2*>());
|
|
CHECK_FALSE(v.try_as<const derived2*>());
|
|
}
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.try_as<const derived*>()->l == 168);
|
|
|
|
CHECK_FALSE(v.try_as<const derived2*>());
|
|
}
|
|
}
|
|
|
|
SUBCASE("derived* to base*") {
|
|
{
|
|
derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK(v.try_as<base2*>()->k == 84);
|
|
CHECK(v.try_as<const base2*>()->k == 84);
|
|
}
|
|
|
|
{
|
|
const derived d{};
|
|
meta::uvalue v{&d};
|
|
CHECK_FALSE(v.try_as<base2*>());
|
|
CHECK(v.try_as<const base2*>()->k == 84);
|
|
}
|
|
}
|
|
}
|