meta.cpp 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683
  1. #include <type_traits>
  2. #include <gtest/gtest.h>
  3. #include <entt/core/hashed_string.hpp>
  4. #include <entt/core/utility.hpp>
  5. #include <entt/meta/factory.hpp>
  6. #include <entt/meta/meta.hpp>
  7. enum class properties {
  8. prop_int,
  9. prop_bool
  10. };
  11. struct empty_type {
  12. virtual ~empty_type() = default;
  13. static void destroy(empty_type *) {
  14. ++counter;
  15. }
  16. inline static int counter = 0;
  17. };
  18. struct fat_type: empty_type {
  19. fat_type() = default;
  20. fat_type(int *value)
  21. : foo{value}, bar{value}
  22. {}
  23. int *foo{nullptr};
  24. int *bar{nullptr};
  25. bool operator==(const fat_type &other) const {
  26. return foo == other.foo && bar == other.bar;
  27. }
  28. };
  29. union union_type {
  30. int i;
  31. double d;
  32. };
  33. bool operator!=(const fat_type &lhs, const fat_type &rhs) {
  34. return !(lhs == rhs);
  35. }
  36. struct base_type {
  37. virtual ~base_type() = default;
  38. };
  39. struct derived_type: base_type {
  40. derived_type() = default;
  41. derived_type(const base_type &, int value, char character)
  42. : i{value}, c{character}
  43. {}
  44. const int i{};
  45. const char c{};
  46. };
  47. derived_type derived_factory(const base_type &, int value) {
  48. return {derived_type{}, value, 'c'};
  49. }
  50. struct data_type {
  51. int i{0};
  52. const int j{1};
  53. inline static int h{2};
  54. inline static const int k{3};
  55. empty_type empty{};
  56. };
  57. struct array_type {
  58. static inline int global[3];
  59. int local[3];
  60. };
  61. struct func_type {
  62. int f(const base_type &, int a, int b) { return f(a, b); }
  63. int f(int a, int b) { value = a; return b*b; }
  64. int f(int v) const { return v*v; }
  65. void g(int v) { value = v*v; }
  66. static int h(int v) { return v; }
  67. static void k(int v) { value = v; }
  68. inline static int value = 0;
  69. };
  70. struct setter_getter_type {
  71. int value{};
  72. int setter(int val) { return value = val; }
  73. int getter() { return value; }
  74. int setter_with_ref(const int &val) { return value = val; }
  75. const int & getter_with_ref() { return value; }
  76. static int static_setter(setter_getter_type *type, int value) { return type->value = value; }
  77. static int static_getter(const setter_getter_type *type) { return type->value; }
  78. };
  79. struct not_comparable_type {
  80. bool operator==(const not_comparable_type &) const = delete;
  81. };
  82. bool operator!=(const not_comparable_type &, const not_comparable_type &) = delete;
  83. struct an_abstract_type {
  84. virtual ~an_abstract_type() = default;
  85. void f(int v) { i = v; }
  86. virtual void g(int) = 0;
  87. int i{};
  88. };
  89. struct another_abstract_type {
  90. virtual ~another_abstract_type() = default;
  91. virtual void h(char) = 0;
  92. char j{};
  93. };
  94. struct concrete_type: an_abstract_type, another_abstract_type {
  95. void f(int v) { i = v*v; } // hide, it's ok :-)
  96. void g(int v) override { i = -v; }
  97. void h(char c) override { j = c; }
  98. };
  99. struct Meta: public ::testing::Test {
  100. static void SetUpTestCase() {
  101. entt::reflect<double>().conv<int>();
  102. entt::reflect<char>("char"_hs, std::make_pair(properties::prop_int, 42));
  103. entt::reflect<properties>()
  104. .data<properties::prop_bool>("prop_bool"_hs)
  105. .data<properties::prop_int>("prop_int"_hs);
  106. entt::reflect<unsigned int>().data<0u>("min"_hs).data<100u>("max"_hs);
  107. entt::reflect<base_type>("base"_hs);
  108. entt::reflect<derived_type>("derived"_hs, std::make_pair(properties::prop_int, 99))
  109. .base<base_type>()
  110. .ctor<const base_type &, int, char>(std::make_pair(properties::prop_bool, false))
  111. .ctor<&derived_factory>(std::make_pair(properties::prop_int, 42));
  112. entt::reflect<empty_type>("empty"_hs)
  113. .dtor<&empty_type::destroy>();
  114. entt::reflect<fat_type>("fat"_hs)
  115. .base<empty_type>()
  116. .dtor<&fat_type::destroy>();
  117. entt::reflect<data_type>("data"_hs)
  118. .data<&data_type::i>("i"_hs, std::make_pair(properties::prop_int, 0))
  119. .data<&data_type::j>("j"_hs, std::make_pair(properties::prop_int, 1))
  120. .data<&data_type::h>("h"_hs, std::make_pair(properties::prop_int, 2))
  121. .data<&data_type::k>("k"_hs, std::make_pair(properties::prop_int, 3))
  122. .data<&data_type::empty>("empty"_hs);
  123. entt::reflect<array_type>("array"_hs)
  124. .data<&array_type::global>("global"_hs)
  125. .data<&array_type::local>("local"_hs);
  126. entt::reflect<func_type>("func"_hs)
  127. .func<entt::overload<int(const base_type &, int, int)>(&func_type::f)>("f3"_hs)
  128. .func<entt::overload<int(int, int)>(&func_type::f)>("f2"_hs, std::make_pair(properties::prop_bool, false))
  129. .func<entt::overload<int(int) const>(&func_type::f)>("f1"_hs, std::make_pair(properties::prop_bool, false))
  130. .func<&func_type::g>("g"_hs, std::make_pair(properties::prop_bool, false))
  131. .func<&func_type::h>("h"_hs, std::make_pair(properties::prop_bool, false))
  132. .func<&func_type::k>("k"_hs, std::make_pair(properties::prop_bool, false));
  133. entt::reflect<setter_getter_type>("setter_getter"_hs)
  134. .data<&setter_getter_type::static_setter, &setter_getter_type::static_getter>("x"_hs)
  135. .data<&setter_getter_type::setter, &setter_getter_type::getter>("y"_hs)
  136. .data<&setter_getter_type::static_setter, &setter_getter_type::getter>("z"_hs)
  137. .data<&setter_getter_type::setter_with_ref, &setter_getter_type::getter_with_ref>("w"_hs);
  138. entt::reflect<an_abstract_type>("an_abstract_type"_hs, std::make_pair(properties::prop_bool, false))
  139. .data<&an_abstract_type::i>("i"_hs)
  140. .func<&an_abstract_type::f>("f"_hs)
  141. .func<&an_abstract_type::g>("g"_hs);
  142. entt::reflect<another_abstract_type>("another_abstract_type"_hs, std::make_pair(properties::prop_int, 42))
  143. .data<&another_abstract_type::j>("j"_hs)
  144. .func<&another_abstract_type::h>("h"_hs);
  145. entt::reflect<concrete_type>("concrete"_hs)
  146. .base<an_abstract_type>()
  147. .base<another_abstract_type>()
  148. .func<&concrete_type::f>("f"_hs);
  149. }
  150. static void SetUpAfterUnregistration() {
  151. entt::reflect<double>().conv<float>();
  152. entt::reflect<derived_type>("my_type"_hs, std::make_pair(properties::prop_bool, false))
  153. .ctor<>();
  154. entt::reflect<another_abstract_type>("your_type"_hs)
  155. .data<&another_abstract_type::j>("a_data_member"_hs)
  156. .func<&another_abstract_type::h>("a_member_function"_hs);
  157. }
  158. void SetUp() override {
  159. empty_type::counter = 0;
  160. func_type::value = 0;
  161. }
  162. };
  163. TEST_F(Meta, Resolve) {
  164. ASSERT_EQ(entt::resolve<derived_type>(), entt::resolve("derived"_hs));
  165. bool found = false;
  166. entt::resolve([&found](auto type) {
  167. found = found || type == entt::resolve<derived_type>();
  168. });
  169. ASSERT_TRUE(found);
  170. }
  171. TEST_F(Meta, MetaAnySBO) {
  172. entt::meta_any any{'c'};
  173. ASSERT_TRUE(any);
  174. ASSERT_FALSE(any.can_cast<void>());
  175. ASSERT_TRUE(any.can_cast<char>());
  176. ASSERT_EQ(any.cast<char>(), 'c');
  177. ASSERT_EQ(std::as_const(any).cast<char>(), 'c');
  178. ASSERT_NE(any.data(), nullptr);
  179. ASSERT_NE(std::as_const(any).data(), nullptr);
  180. ASSERT_EQ(any, entt::meta_any{'c'});
  181. ASSERT_NE(any, entt::meta_any{'h'});
  182. }
  183. TEST_F(Meta, MetaAnyNoSBO) {
  184. int value = 42;
  185. fat_type instance{&value};
  186. entt::meta_any any{instance};
  187. ASSERT_TRUE(any);
  188. ASSERT_FALSE(any.can_cast<void>());
  189. ASSERT_TRUE(any.can_cast<fat_type>());
  190. ASSERT_EQ(any.cast<fat_type>(), instance);
  191. ASSERT_EQ(std::as_const(any).cast<fat_type>(), instance);
  192. ASSERT_NE(any.data(), nullptr);
  193. ASSERT_NE(std::as_const(any).data(), nullptr);
  194. ASSERT_EQ(any, entt::meta_any{instance});
  195. ASSERT_NE(any, fat_type{});
  196. }
  197. TEST_F(Meta, MetaAnyEmpty) {
  198. entt::meta_any any{};
  199. ASSERT_FALSE(any);
  200. ASSERT_FALSE(any.type());
  201. ASSERT_FALSE(any.can_cast<void>());
  202. ASSERT_FALSE(any.can_cast<empty_type>());
  203. ASSERT_EQ(any.data(), nullptr);
  204. ASSERT_EQ(std::as_const(any).data(), nullptr);
  205. ASSERT_EQ(any, entt::meta_any{});
  206. ASSERT_NE(any, entt::meta_any{'c'});
  207. }
  208. TEST_F(Meta, MetaAnySBOInPlaceConstruction) {
  209. entt::meta_any any{std::in_place_type<int>, 42};
  210. ASSERT_TRUE(any);
  211. ASSERT_FALSE(any.can_cast<void>());
  212. ASSERT_TRUE(any.can_cast<int>());
  213. ASSERT_EQ(any.cast<int>(), 42);
  214. ASSERT_EQ(std::as_const(any).cast<int>(), 42);
  215. ASSERT_NE(any.data(), nullptr);
  216. ASSERT_NE(std::as_const(any).data(), nullptr);
  217. ASSERT_EQ(any, (entt::meta_any{std::in_place_type<int>, 42}));
  218. ASSERT_EQ(any, entt::meta_any{42});
  219. ASSERT_NE(any, entt::meta_any{3});
  220. }
  221. TEST_F(Meta, MetaAnySBOCopyConstruction) {
  222. entt::meta_any any{42};
  223. entt::meta_any other{any};
  224. ASSERT_TRUE(any);
  225. ASSERT_TRUE(other);
  226. ASSERT_FALSE(other.can_cast<void>());
  227. ASSERT_TRUE(other.can_cast<int>());
  228. ASSERT_EQ(other.cast<int>(), 42);
  229. ASSERT_EQ(std::as_const(other).cast<int>(), 42);
  230. ASSERT_EQ(other, entt::meta_any{42});
  231. ASSERT_NE(other, entt::meta_any{0});
  232. }
  233. TEST_F(Meta, MetaAnySBOCopyAssignment) {
  234. entt::meta_any any{42};
  235. entt::meta_any other{3};
  236. other = any;
  237. ASSERT_TRUE(any);
  238. ASSERT_TRUE(other);
  239. ASSERT_FALSE(other.can_cast<void>());
  240. ASSERT_TRUE(other.can_cast<int>());
  241. ASSERT_EQ(other.cast<int>(), 42);
  242. ASSERT_EQ(std::as_const(other).cast<int>(), 42);
  243. ASSERT_EQ(other, entt::meta_any{42});
  244. ASSERT_NE(other, entt::meta_any{0});
  245. }
  246. TEST_F(Meta, MetaAnySBOMoveConstruction) {
  247. entt::meta_any any{42};
  248. entt::meta_any other{std::move(any)};
  249. ASSERT_FALSE(any);
  250. ASSERT_TRUE(other);
  251. ASSERT_FALSE(other.can_cast<void>());
  252. ASSERT_TRUE(other.can_cast<int>());
  253. ASSERT_EQ(other.cast<int>(), 42);
  254. ASSERT_EQ(std::as_const(other).cast<int>(), 42);
  255. ASSERT_EQ(other, entt::meta_any{42});
  256. ASSERT_NE(other, entt::meta_any{0});
  257. }
  258. TEST_F(Meta, MetaAnySBOMoveAssignment) {
  259. entt::meta_any any{42};
  260. entt::meta_any other{3};
  261. other = std::move(any);
  262. ASSERT_FALSE(any);
  263. ASSERT_TRUE(other);
  264. ASSERT_FALSE(other.can_cast<void>());
  265. ASSERT_TRUE(other.can_cast<int>());
  266. ASSERT_EQ(other.cast<int>(), 42);
  267. ASSERT_EQ(std::as_const(other).cast<int>(), 42);
  268. ASSERT_EQ(other, entt::meta_any{42});
  269. ASSERT_NE(other, entt::meta_any{0});
  270. }
  271. TEST_F(Meta, MetaAnyNoSBOInPlaceConstruction) {
  272. int value = 42;
  273. fat_type instance{&value};
  274. entt::meta_any any{std::in_place_type<fat_type>, instance};
  275. ASSERT_TRUE(any);
  276. ASSERT_FALSE(any.can_cast<void>());
  277. ASSERT_TRUE(any.can_cast<fat_type>());
  278. ASSERT_EQ(any.cast<fat_type>(), instance);
  279. ASSERT_EQ(std::as_const(any).cast<fat_type>(), instance);
  280. ASSERT_NE(any.data(), nullptr);
  281. ASSERT_NE(std::as_const(any).data(), nullptr);
  282. ASSERT_EQ(any, (entt::meta_any{std::in_place_type<fat_type>, instance}));
  283. ASSERT_EQ(any, entt::meta_any{instance});
  284. ASSERT_NE(any, entt::meta_any{fat_type{}});
  285. }
  286. TEST_F(Meta, MetaAnyNoSBOCopyConstruction) {
  287. int value = 42;
  288. fat_type instance{&value};
  289. entt::meta_any any{instance};
  290. entt::meta_any other{any};
  291. ASSERT_TRUE(any);
  292. ASSERT_TRUE(other);
  293. ASSERT_FALSE(other.can_cast<void>());
  294. ASSERT_TRUE(other.can_cast<fat_type>());
  295. ASSERT_EQ(other.cast<fat_type>(), instance);
  296. ASSERT_EQ(std::as_const(other).cast<fat_type>(), instance);
  297. ASSERT_EQ(other, entt::meta_any{instance});
  298. ASSERT_NE(other, fat_type{});
  299. }
  300. TEST_F(Meta, MetaAnyNoSBOCopyAssignment) {
  301. int value = 42;
  302. fat_type instance{&value};
  303. entt::meta_any any{instance};
  304. entt::meta_any other{3};
  305. other = any;
  306. ASSERT_TRUE(any);
  307. ASSERT_TRUE(other);
  308. ASSERT_FALSE(other.can_cast<void>());
  309. ASSERT_TRUE(other.can_cast<fat_type>());
  310. ASSERT_EQ(other.cast<fat_type>(), instance);
  311. ASSERT_EQ(std::as_const(other).cast<fat_type>(), instance);
  312. ASSERT_EQ(other, entt::meta_any{instance});
  313. ASSERT_NE(other, fat_type{});
  314. }
  315. TEST_F(Meta, MetaAnyNoSBOMoveConstruction) {
  316. int value = 42;
  317. fat_type instance{&value};
  318. entt::meta_any any{instance};
  319. entt::meta_any other{std::move(any)};
  320. ASSERT_FALSE(any);
  321. ASSERT_TRUE(other);
  322. ASSERT_FALSE(other.can_cast<void>());
  323. ASSERT_TRUE(other.can_cast<fat_type>());
  324. ASSERT_EQ(other.cast<fat_type>(), instance);
  325. ASSERT_EQ(std::as_const(other).cast<fat_type>(), instance);
  326. ASSERT_EQ(other, entt::meta_any{instance});
  327. ASSERT_NE(other, fat_type{});
  328. }
  329. TEST_F(Meta, MetaAnyNoSBOMoveAssignment) {
  330. int value = 42;
  331. fat_type instance{&value};
  332. entt::meta_any any{instance};
  333. entt::meta_any other{3};
  334. other = std::move(any);
  335. ASSERT_FALSE(any);
  336. ASSERT_TRUE(other);
  337. ASSERT_FALSE(other.can_cast<void>());
  338. ASSERT_TRUE(other.can_cast<fat_type>());
  339. ASSERT_EQ(other.cast<fat_type>(), instance);
  340. ASSERT_EQ(std::as_const(other).cast<fat_type>(), instance);
  341. ASSERT_EQ(other, entt::meta_any{instance});
  342. ASSERT_NE(other, fat_type{});
  343. }
  344. TEST_F(Meta, MetaAnySBODestruction) {
  345. ASSERT_EQ(empty_type::counter, 0);
  346. { entt::meta_any any{empty_type{}}; }
  347. ASSERT_EQ(empty_type::counter, 1);
  348. }
  349. TEST_F(Meta, MetaAnyNoSBODestruction) {
  350. ASSERT_EQ(fat_type::counter, 0);
  351. { entt::meta_any any{fat_type{}}; }
  352. ASSERT_EQ(fat_type::counter, 1);
  353. }
  354. TEST_F(Meta, MetaAnyEmplace) {
  355. entt::meta_any any{};
  356. any.emplace<int>(42);
  357. ASSERT_TRUE(any);
  358. ASSERT_FALSE(any.can_cast<void>());
  359. ASSERT_TRUE(any.can_cast<int>());
  360. ASSERT_EQ(any.cast<int>(), 42);
  361. ASSERT_EQ(std::as_const(any).cast<int>(), 42);
  362. ASSERT_NE(any.data(), nullptr);
  363. ASSERT_NE(std::as_const(any).data(), nullptr);
  364. ASSERT_EQ(any, (entt::meta_any{std::in_place_type<int>, 42}));
  365. ASSERT_EQ(any, entt::meta_any{42});
  366. ASSERT_NE(any, entt::meta_any{3});
  367. }
  368. TEST_F(Meta, MetaAnySBOSwap) {
  369. entt::meta_any lhs{'c'};
  370. entt::meta_any rhs{42};
  371. std::swap(lhs, rhs);
  372. ASSERT_TRUE(lhs.can_cast<int>());
  373. ASSERT_EQ(lhs.cast<int>(), 42);
  374. ASSERT_TRUE(rhs.can_cast<char>());
  375. ASSERT_EQ(rhs.cast<char>(), 'c');
  376. }
  377. TEST_F(Meta, MetaAnyNoSBOSwap) {
  378. int i, j;
  379. entt::meta_any lhs{fat_type{&i}};
  380. entt::meta_any rhs{fat_type{&j}};
  381. std::swap(lhs, rhs);
  382. ASSERT_EQ(lhs.cast<fat_type>().foo, &j);
  383. ASSERT_EQ(rhs.cast<fat_type>().bar, &i);
  384. }
  385. TEST_F(Meta, MetaAnySBOWithNoSBOSwap) {
  386. int value = 42;
  387. entt::meta_any lhs{fat_type{&value}};
  388. entt::meta_any rhs{'c'};
  389. std::swap(lhs, rhs);
  390. ASSERT_TRUE(lhs.can_cast<char>());
  391. ASSERT_EQ(lhs.cast<char>(), 'c');
  392. ASSERT_TRUE(rhs.can_cast<fat_type>());
  393. ASSERT_EQ(rhs.cast<fat_type>().foo, &value);
  394. ASSERT_EQ(rhs.cast<fat_type>().bar, &value);
  395. }
  396. TEST_F(Meta, MetaAnySBOWithEmptySwap) {
  397. entt::meta_any lhs{'c'};
  398. entt::meta_any rhs{};
  399. std::swap(lhs, rhs);
  400. ASSERT_FALSE(lhs);
  401. ASSERT_TRUE(rhs.can_cast<char>());
  402. ASSERT_EQ(rhs.cast<char>(), 'c');
  403. std::swap(lhs, rhs);
  404. ASSERT_FALSE(rhs);
  405. ASSERT_TRUE(lhs.can_cast<char>());
  406. ASSERT_EQ(lhs.cast<char>(), 'c');
  407. }
  408. TEST_F(Meta, MetaAnyNoSBOWithEmptySwap) {
  409. int i;
  410. entt::meta_any lhs{fat_type{&i}};
  411. entt::meta_any rhs{};
  412. std::swap(lhs, rhs);
  413. ASSERT_EQ(rhs.cast<fat_type>().bar, &i);
  414. std::swap(lhs, rhs);
  415. ASSERT_EQ(lhs.cast<fat_type>().bar, &i);
  416. }
  417. TEST_F(Meta, MetaAnyComparable) {
  418. entt::meta_any any{'c'};
  419. ASSERT_EQ(any, any);
  420. ASSERT_EQ(any, entt::meta_any{'c'});
  421. ASSERT_NE(any, entt::meta_any{'a'});
  422. ASSERT_NE(any, entt::meta_any{});
  423. ASSERT_TRUE(any == any);
  424. ASSERT_TRUE(any == entt::meta_any{'c'});
  425. ASSERT_FALSE(any == entt::meta_any{'a'});
  426. ASSERT_TRUE(any != entt::meta_any{'a'});
  427. ASSERT_TRUE(any != entt::meta_any{});
  428. }
  429. TEST_F(Meta, MetaAnyNotComparable) {
  430. entt::meta_any any{not_comparable_type{}};
  431. ASSERT_EQ(any, any);
  432. ASSERT_NE(any, entt::meta_any{not_comparable_type{}});
  433. ASSERT_NE(any, entt::meta_any{});
  434. ASSERT_TRUE(any == any);
  435. ASSERT_FALSE(any == entt::meta_any{not_comparable_type{}});
  436. ASSERT_TRUE(any != entt::meta_any{});
  437. }
  438. TEST_F(Meta, MetaAnyCast) {
  439. entt::meta_any any{derived_type{}};
  440. entt::meta_handle handle{any};
  441. ASSERT_TRUE(any);
  442. ASSERT_EQ(any.type(), entt::resolve<derived_type>());
  443. ASSERT_FALSE(any.can_cast<void>());
  444. ASSERT_TRUE(any.can_cast<base_type>());
  445. ASSERT_TRUE(any.can_cast<derived_type>());
  446. ASSERT_EQ(&any.cast<base_type>(), handle.try_cast<base_type>());
  447. ASSERT_EQ(&any.cast<derived_type>(), handle.try_cast<derived_type>());
  448. ASSERT_EQ(&std::as_const(any).cast<base_type>(), handle.try_cast<base_type>());
  449. ASSERT_EQ(&std::as_const(any).cast<derived_type>(), handle.try_cast<derived_type>());
  450. }
  451. TEST_F(Meta, MetaAnyConvert) {
  452. entt::meta_any any{42.};
  453. ASSERT_TRUE(any);
  454. ASSERT_EQ(any.type(), entt::resolve<double>());
  455. ASSERT_FALSE(any.can_convert<char>());
  456. ASSERT_TRUE(any.can_convert<double>());
  457. ASSERT_TRUE(any.can_convert<int>());
  458. ASSERT_TRUE(any.convert<double>());
  459. ASSERT_FALSE(any.convert<char>());
  460. ASSERT_EQ(any.type(), entt::resolve<double>());
  461. ASSERT_EQ(any.cast<double>(), 42.);
  462. ASSERT_TRUE(any.convert<int>());
  463. ASSERT_EQ(any.type(), entt::resolve<int>());
  464. ASSERT_EQ(any.cast<int>(), 42);
  465. }
  466. TEST_F(Meta, MetaAnyConstConvert) {
  467. const entt::meta_any any{42.};
  468. ASSERT_TRUE(any);
  469. ASSERT_EQ(any.type(), entt::resolve<double>());
  470. ASSERT_FALSE(any.can_convert<char>());
  471. ASSERT_TRUE(any.can_convert<double>());
  472. ASSERT_TRUE(any.can_convert<int>());
  473. ASSERT_TRUE(any.convert<double>());
  474. ASSERT_FALSE(any.convert<char>());
  475. ASSERT_EQ(any.type(), entt::resolve<double>());
  476. ASSERT_EQ(any.cast<double>(), 42.);
  477. auto other = any.convert<int>();
  478. ASSERT_EQ(any.type(), entt::resolve<double>());
  479. ASSERT_EQ(any.cast<double>(), 42.);
  480. ASSERT_EQ(other.type(), entt::resolve<int>());
  481. ASSERT_EQ(other.cast<int>(), 42);
  482. }
  483. TEST_F(Meta, MetaHandleFromObject) {
  484. empty_type empty{};
  485. entt::meta_handle handle{empty};
  486. ASSERT_TRUE(handle);
  487. ASSERT_EQ(handle.type(), entt::resolve<empty_type>());
  488. ASSERT_EQ(handle.try_cast<void>(), nullptr);
  489. ASSERT_EQ(handle.try_cast<empty_type>(), &empty);
  490. ASSERT_EQ(std::as_const(handle).try_cast<empty_type>(), &empty);
  491. ASSERT_EQ(handle.data(), &empty);
  492. ASSERT_EQ(std::as_const(handle).data(), &empty);
  493. }
  494. TEST_F(Meta, MetaHandleFromMetaAny) {
  495. entt::meta_any any{42};
  496. entt::meta_handle handle{any};
  497. ASSERT_TRUE(handle);
  498. ASSERT_EQ(handle.type(), entt::resolve<int>());
  499. ASSERT_EQ(handle.try_cast<void>(), nullptr);
  500. ASSERT_EQ(handle.try_cast<int>(), any.data());
  501. ASSERT_EQ(std::as_const(handle).try_cast<int>(), any.data());
  502. ASSERT_EQ(handle.data(), any.data());
  503. ASSERT_EQ(std::as_const(handle).data(), any.data());
  504. }
  505. TEST_F(Meta, MetaHandleEmpty) {
  506. entt::meta_handle handle{};
  507. ASSERT_FALSE(handle);
  508. ASSERT_FALSE(handle.type());
  509. ASSERT_EQ(handle.try_cast<void>(), nullptr);
  510. ASSERT_EQ(handle.try_cast<empty_type>(), nullptr);
  511. ASSERT_EQ(handle.data(), nullptr);
  512. ASSERT_EQ(std::as_const(handle).data(), nullptr);
  513. }
  514. TEST_F(Meta, MetaHandleTryCast) {
  515. derived_type derived{};
  516. base_type *base = &derived;
  517. entt::meta_handle handle{derived};
  518. ASSERT_TRUE(handle);
  519. ASSERT_EQ(handle.type(), entt::resolve<derived_type>());
  520. ASSERT_EQ(handle.try_cast<void>(), nullptr);
  521. ASSERT_EQ(handle.try_cast<base_type>(), base);
  522. ASSERT_EQ(handle.try_cast<derived_type>(), &derived);
  523. ASSERT_EQ(std::as_const(handle).try_cast<base_type>(), base);
  524. ASSERT_EQ(std::as_const(handle).try_cast<derived_type>(), &derived);
  525. ASSERT_EQ(handle.data(), &derived);
  526. ASSERT_EQ(std::as_const(handle).data(), &derived);
  527. }
  528. TEST_F(Meta, MetaProp) {
  529. auto prop = entt::resolve<char>().prop(properties::prop_int);
  530. ASSERT_TRUE(prop);
  531. ASSERT_NE(prop, entt::meta_prop{});
  532. ASSERT_EQ(prop.key(), properties::prop_int);
  533. ASSERT_EQ(prop.value(), 42);
  534. }
  535. TEST_F(Meta, MetaBase) {
  536. auto base = entt::resolve<derived_type>().base("base"_hs);
  537. derived_type derived{};
  538. ASSERT_TRUE(base);
  539. ASSERT_NE(base, entt::meta_base{});
  540. ASSERT_EQ(base.parent(), entt::resolve("derived"_hs));
  541. ASSERT_EQ(base.type(), entt::resolve<base_type>());
  542. ASSERT_EQ(base.cast(&derived), static_cast<base_type *>(&derived));
  543. }
  544. TEST_F(Meta, MetaConv) {
  545. auto conv = entt::resolve<double>().conv<int>();
  546. double value = 3.;
  547. ASSERT_TRUE(conv);
  548. ASSERT_NE(conv, entt::meta_conv{});
  549. ASSERT_EQ(conv.parent(), entt::resolve<double>());
  550. ASSERT_EQ(conv.type(), entt::resolve<int>());
  551. auto any = conv.convert(&value);
  552. ASSERT_TRUE(any);
  553. ASSERT_EQ(any.type(), entt::resolve<int>());
  554. ASSERT_EQ(any.cast<int>(), 3);
  555. }
  556. TEST_F(Meta, MetaCtor) {
  557. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int, char>();
  558. ASSERT_TRUE(ctor);
  559. ASSERT_NE(ctor, entt::meta_ctor{});
  560. ASSERT_EQ(ctor.parent(), entt::resolve("derived"_hs));
  561. ASSERT_EQ(ctor.size(), entt::meta_ctor::size_type{3});
  562. ASSERT_EQ(ctor.arg(entt::meta_ctor::size_type{0}), entt::resolve<base_type>());
  563. ASSERT_EQ(ctor.arg(entt::meta_ctor::size_type{1}), entt::resolve<int>());
  564. ASSERT_EQ(ctor.arg(entt::meta_ctor::size_type{2}), entt::resolve<char>());
  565. ASSERT_FALSE(ctor.arg(entt::meta_ctor::size_type{3}));
  566. auto any = ctor.invoke(base_type{}, 42, 'c');
  567. auto empty = ctor.invoke();
  568. ASSERT_FALSE(empty);
  569. ASSERT_TRUE(any);
  570. ASSERT_TRUE(any.can_cast<derived_type>());
  571. ASSERT_EQ(any.cast<derived_type>().i, 42);
  572. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  573. ctor.prop([](auto prop) {
  574. ASSERT_TRUE(prop);
  575. ASSERT_EQ(prop.key(), properties::prop_bool);
  576. ASSERT_FALSE(prop.value().template cast<bool>());
  577. });
  578. ASSERT_FALSE(ctor.prop(properties::prop_int));
  579. auto prop = ctor.prop(properties::prop_bool);
  580. ASSERT_TRUE(prop);
  581. ASSERT_EQ(prop.key(), properties::prop_bool);
  582. ASSERT_FALSE(prop.value().template cast<bool>());
  583. }
  584. TEST_F(Meta, MetaCtorFunc) {
  585. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int>();
  586. ASSERT_TRUE(ctor);
  587. ASSERT_EQ(ctor.parent(), entt::resolve("derived"_hs));
  588. ASSERT_EQ(ctor.size(), entt::meta_ctor::size_type{2});
  589. ASSERT_EQ(ctor.arg(entt::meta_ctor::size_type{0}), entt::resolve<base_type>());
  590. ASSERT_EQ(ctor.arg(entt::meta_ctor::size_type{1}), entt::resolve<int>());
  591. ASSERT_FALSE(ctor.arg(entt::meta_ctor::size_type{2}));
  592. auto any = ctor.invoke(derived_type{}, 42);
  593. auto empty = ctor.invoke(3, 'c');
  594. ASSERT_FALSE(empty);
  595. ASSERT_TRUE(any);
  596. ASSERT_TRUE(any.can_cast<derived_type>());
  597. ASSERT_EQ(any.cast<derived_type>().i, 42);
  598. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  599. ctor.prop([](auto prop) {
  600. ASSERT_TRUE(prop);
  601. ASSERT_EQ(prop.key(), properties::prop_int);
  602. ASSERT_EQ(prop.value(), 42);
  603. });
  604. ASSERT_FALSE(ctor.prop(properties::prop_bool));
  605. auto prop = ctor.prop(properties::prop_int);
  606. ASSERT_TRUE(prop);
  607. ASSERT_EQ(prop.key(), properties::prop_int);
  608. ASSERT_EQ(prop.value(), 42);
  609. }
  610. TEST_F(Meta, MetaCtorMetaAnyArgs) {
  611. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int, char>();
  612. auto any = ctor.invoke(base_type{}, entt::meta_any{42}, entt::meta_any{'c'});
  613. ASSERT_TRUE(any);
  614. ASSERT_TRUE(any.can_cast<derived_type>());
  615. ASSERT_EQ(any.cast<derived_type>().i, 42);
  616. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  617. }
  618. TEST_F(Meta, MetaCtorInvalidArgs) {
  619. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int, char>();
  620. ASSERT_FALSE(ctor.invoke(base_type{}, entt::meta_any{'c'}, entt::meta_any{42}));
  621. }
  622. TEST_F(Meta, MetaCtorCastAndConvert) {
  623. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int, char>();
  624. auto any = ctor.invoke(entt::meta_any{derived_type{}}, entt::meta_any{42.}, entt::meta_any{'c'});
  625. ASSERT_TRUE(any);
  626. ASSERT_TRUE(any.can_cast<derived_type>());
  627. ASSERT_EQ(any.cast<derived_type>().i, 42);
  628. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  629. }
  630. TEST_F(Meta, MetaCtorFuncMetaAnyArgs) {
  631. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int>();
  632. auto any = ctor.invoke(base_type{}, entt::meta_any{42});
  633. ASSERT_TRUE(any);
  634. ASSERT_TRUE(any.can_cast<derived_type>());
  635. ASSERT_EQ(any.cast<derived_type>().i, 42);
  636. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  637. }
  638. TEST_F(Meta, MetaCtorFuncInvalidArgs) {
  639. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int>();
  640. ASSERT_FALSE(ctor.invoke(base_type{}, entt::meta_any{'c'}));
  641. }
  642. TEST_F(Meta, MetaCtorFuncCastAndConvert) {
  643. auto ctor = entt::resolve<derived_type>().ctor<const base_type &, int>();
  644. auto any = ctor.invoke(entt::meta_any{derived_type{}}, entt::meta_any{42.});
  645. ASSERT_TRUE(any);
  646. ASSERT_TRUE(any.can_cast<derived_type>());
  647. ASSERT_EQ(any.cast<derived_type>().i, 42);
  648. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  649. }
  650. TEST_F(Meta, MetaDtor) {
  651. auto dtor = entt::resolve<empty_type>().dtor();
  652. empty_type empty{};
  653. ASSERT_TRUE(dtor);
  654. ASSERT_NE(dtor, entt::meta_dtor{});
  655. ASSERT_EQ(dtor.parent(), entt::resolve("empty"_hs));
  656. ASSERT_EQ(empty_type::counter, 0);
  657. ASSERT_TRUE(dtor.invoke(empty));
  658. ASSERT_EQ(empty_type::counter, 1);
  659. }
  660. TEST_F(Meta, MetaDtorMetaAnyArg) {
  661. auto dtor = entt::resolve<empty_type>().dtor();
  662. entt::meta_any any{empty_type{}};
  663. ASSERT_EQ(empty_type::counter, 0);
  664. ASSERT_TRUE(dtor.invoke(any));
  665. ASSERT_EQ(empty_type::counter, 1);
  666. }
  667. TEST_F(Meta, MetaDtorMetaAnyInvalidArg) {
  668. ASSERT_FALSE(entt::resolve<empty_type>().dtor().invoke(int{}));
  669. }
  670. TEST_F(Meta, MetaData) {
  671. auto data = entt::resolve<data_type>().data("i"_hs);
  672. data_type instance{};
  673. ASSERT_TRUE(data);
  674. ASSERT_NE(data, entt::meta_data{});
  675. ASSERT_EQ(data.parent(), entt::resolve("data"_hs));
  676. ASSERT_EQ(data.type(), entt::resolve<int>());
  677. ASSERT_EQ(data.identifier(), "i"_hs);
  678. ASSERT_FALSE(data.is_const());
  679. ASSERT_FALSE(data.is_static());
  680. ASSERT_EQ(data.get(instance).cast<int>(), 0);
  681. ASSERT_TRUE(data.set(instance, 42));
  682. ASSERT_EQ(data.get(instance).cast<int>(), 42);
  683. data.prop([](auto prop) {
  684. ASSERT_TRUE(prop);
  685. ASSERT_EQ(prop.key(), properties::prop_int);
  686. ASSERT_EQ(prop.value(), 0);
  687. });
  688. ASSERT_FALSE(data.prop(properties::prop_bool));
  689. auto prop = data.prop(properties::prop_int);
  690. ASSERT_TRUE(prop);
  691. ASSERT_EQ(prop.key(), properties::prop_int);
  692. ASSERT_EQ(prop.value(), 0);
  693. }
  694. TEST_F(Meta, MetaDataConst) {
  695. auto data = entt::resolve<data_type>().data("j"_hs);
  696. data_type instance{};
  697. ASSERT_TRUE(data);
  698. ASSERT_EQ(data.parent(), entt::resolve("data"_hs));
  699. ASSERT_EQ(data.type(), entt::resolve<int>());
  700. ASSERT_EQ(data.identifier(), "j"_hs);
  701. ASSERT_TRUE(data.is_const());
  702. ASSERT_FALSE(data.is_static());
  703. ASSERT_EQ(data.get(instance).cast<int>(), 1);
  704. ASSERT_FALSE(data.set(instance, 42));
  705. ASSERT_EQ(data.get(instance).cast<int>(), 1);
  706. data.prop([](auto prop) {
  707. ASSERT_TRUE(prop);
  708. ASSERT_EQ(prop.key(), properties::prop_int);
  709. ASSERT_EQ(prop.value(), 1);
  710. });
  711. ASSERT_FALSE(data.prop(properties::prop_bool));
  712. auto prop = data.prop(properties::prop_int);
  713. ASSERT_TRUE(prop);
  714. ASSERT_EQ(prop.key(), properties::prop_int);
  715. ASSERT_EQ(prop.value(), 1);
  716. }
  717. TEST_F(Meta, MetaDataStatic) {
  718. auto data = entt::resolve<data_type>().data("h"_hs);
  719. ASSERT_TRUE(data);
  720. ASSERT_EQ(data.parent(), entt::resolve("data"_hs));
  721. ASSERT_EQ(data.type(), entt::resolve<int>());
  722. ASSERT_EQ(data.identifier(), "h"_hs);
  723. ASSERT_FALSE(data.is_const());
  724. ASSERT_TRUE(data.is_static());
  725. ASSERT_EQ(data.get({}).cast<int>(), 2);
  726. ASSERT_TRUE(data.set({}, 42));
  727. ASSERT_EQ(data.get({}).cast<int>(), 42);
  728. data.prop([](auto prop) {
  729. ASSERT_TRUE(prop);
  730. ASSERT_EQ(prop.key(), properties::prop_int);
  731. ASSERT_EQ(prop.value(), 2);
  732. });
  733. ASSERT_FALSE(data.prop(properties::prop_bool));
  734. auto prop = data.prop(properties::prop_int);
  735. ASSERT_TRUE(prop);
  736. ASSERT_EQ(prop.key(), properties::prop_int);
  737. ASSERT_EQ(prop.value(), 2);
  738. }
  739. TEST_F(Meta, MetaDataConstStatic) {
  740. auto data = entt::resolve<data_type>().data("k"_hs);
  741. ASSERT_TRUE(data);
  742. ASSERT_EQ(data.parent(), entt::resolve("data"_hs));
  743. ASSERT_EQ(data.type(), entt::resolve<int>());
  744. ASSERT_EQ(data.identifier(), "k"_hs);
  745. ASSERT_TRUE(data.is_const());
  746. ASSERT_TRUE(data.is_static());
  747. ASSERT_EQ(data.get({}).cast<int>(), 3);
  748. ASSERT_FALSE(data.set({}, 42));
  749. ASSERT_EQ(data.get({}).cast<int>(), 3);
  750. data.prop([](auto prop) {
  751. ASSERT_TRUE(prop);
  752. ASSERT_EQ(prop.key(), properties::prop_int);
  753. ASSERT_EQ(prop.value(), 3);
  754. });
  755. ASSERT_FALSE(data.prop(properties::prop_bool));
  756. auto prop = data.prop(properties::prop_int);
  757. ASSERT_TRUE(prop);
  758. ASSERT_EQ(prop.key(), properties::prop_int);
  759. ASSERT_EQ(prop.value(), 3);
  760. }
  761. TEST_F(Meta, MetaDataGetMetaAnyArg) {
  762. auto data = entt::resolve<data_type>().data("i"_hs);
  763. entt::meta_any any{data_type{}};
  764. any.cast<data_type>().i = 99;
  765. const auto value = data.get(any);
  766. ASSERT_TRUE(value);
  767. ASSERT_TRUE(value.can_cast<int>());
  768. ASSERT_EQ(value.cast<int>(), 99);
  769. }
  770. TEST_F(Meta, MetaDataGetInvalidArg) {
  771. ASSERT_FALSE(entt::resolve<data_type>().data("i"_hs).get(0));
  772. }
  773. TEST_F(Meta, MetaDataSetMetaAnyArg) {
  774. auto data = entt::resolve<data_type>().data("i"_hs);
  775. entt::meta_any any{data_type{}};
  776. entt::meta_any value{42};
  777. ASSERT_EQ(any.cast<data_type>().i, 0);
  778. ASSERT_TRUE(data.set(any, value));
  779. ASSERT_EQ(any.cast<data_type>().i, 42);
  780. }
  781. TEST_F(Meta, MetaDataSetInvalidArg) {
  782. ASSERT_FALSE(entt::resolve<data_type>().data("i"_hs).set({}, 'c'));
  783. }
  784. TEST_F(Meta, MetaDataSetCast) {
  785. auto data = entt::resolve<data_type>().data("empty"_hs);
  786. data_type instance{};
  787. ASSERT_EQ(empty_type::counter, 0);
  788. ASSERT_TRUE(data.set(instance, fat_type{}));
  789. ASSERT_EQ(empty_type::counter, 1);
  790. }
  791. TEST_F(Meta, MetaDataSetConvert) {
  792. auto data = entt::resolve<data_type>().data("i"_hs);
  793. data_type instance{};
  794. ASSERT_EQ(instance.i, 0);
  795. ASSERT_TRUE(data.set(instance, 3.));
  796. ASSERT_EQ(instance.i, 3);
  797. }
  798. TEST_F(Meta, MetaDataSetterGetterAsFreeFunctions) {
  799. auto data = entt::resolve<setter_getter_type>().data("x"_hs);
  800. setter_getter_type instance{};
  801. ASSERT_TRUE(data);
  802. ASSERT_NE(data, entt::meta_data{});
  803. ASSERT_EQ(data.parent(), entt::resolve("setter_getter"_hs));
  804. ASSERT_EQ(data.type(), entt::resolve<int>());
  805. ASSERT_EQ(data.identifier(), "x"_hs);
  806. ASSERT_FALSE(data.is_const());
  807. ASSERT_FALSE(data.is_static());
  808. ASSERT_EQ(data.get(instance).cast<int>(), 0);
  809. ASSERT_TRUE(data.set(instance, 42));
  810. ASSERT_EQ(data.get(instance).cast<int>(), 42);
  811. }
  812. TEST_F(Meta, MetaDataSetterGetterAsMemberFunctions) {
  813. auto data = entt::resolve<setter_getter_type>().data("y"_hs);
  814. setter_getter_type instance{};
  815. ASSERT_TRUE(data);
  816. ASSERT_NE(data, entt::meta_data{});
  817. ASSERT_EQ(data.parent(), entt::resolve("setter_getter"_hs));
  818. ASSERT_EQ(data.type(), entt::resolve<int>());
  819. ASSERT_EQ(data.identifier(), "y"_hs);
  820. ASSERT_FALSE(data.is_const());
  821. ASSERT_FALSE(data.is_static());
  822. ASSERT_EQ(data.get(instance).cast<int>(), 0);
  823. ASSERT_TRUE(data.set(instance, 42));
  824. ASSERT_EQ(data.get(instance).cast<int>(), 42);
  825. }
  826. TEST_F(Meta, MetaDataSetterGetterWithRefAsMemberFunctions) {
  827. auto data = entt::resolve<setter_getter_type>().data("w"_hs);
  828. setter_getter_type instance{};
  829. ASSERT_TRUE(data);
  830. ASSERT_NE(data, entt::meta_data{});
  831. ASSERT_EQ(data.parent(), entt::resolve("setter_getter"_hs));
  832. ASSERT_EQ(data.type(), entt::resolve<int>());
  833. ASSERT_EQ(data.identifier(), "w"_hs);
  834. ASSERT_FALSE(data.is_const());
  835. ASSERT_FALSE(data.is_static());
  836. ASSERT_EQ(data.get(instance).cast<int>(), 0);
  837. ASSERT_TRUE(data.set(instance, 42));
  838. ASSERT_EQ(data.get(instance).cast<int>(), 42);
  839. }
  840. TEST_F(Meta, MetaDataSetterGetterMixed) {
  841. auto data = entt::resolve<setter_getter_type>().data("z"_hs);
  842. setter_getter_type instance{};
  843. ASSERT_TRUE(data);
  844. ASSERT_NE(data, entt::meta_data{});
  845. ASSERT_EQ(data.parent(), entt::resolve("setter_getter"_hs));
  846. ASSERT_EQ(data.type(), entt::resolve<int>());
  847. ASSERT_EQ(data.identifier(), "z"_hs);
  848. ASSERT_FALSE(data.is_const());
  849. ASSERT_FALSE(data.is_static());
  850. ASSERT_EQ(data.get(instance).cast<int>(), 0);
  851. ASSERT_TRUE(data.set(instance, 42));
  852. ASSERT_EQ(data.get(instance).cast<int>(), 42);
  853. }
  854. TEST_F(Meta, MetaDataArrayStatic) {
  855. auto data = entt::resolve<array_type>().data("global"_hs);
  856. array_type::global[0] = 3;
  857. array_type::global[1] = 5;
  858. array_type::global[2] = 7;
  859. ASSERT_TRUE(data);
  860. ASSERT_NE(data, entt::meta_data{});
  861. ASSERT_EQ(data.parent(), entt::resolve("array"_hs));
  862. ASSERT_EQ(data.type(), entt::resolve<int[3]>());
  863. ASSERT_EQ(data.identifier(), "global"_hs);
  864. ASSERT_FALSE(data.is_const());
  865. ASSERT_TRUE(data.is_static());
  866. ASSERT_TRUE(data.type().is_array());
  867. ASSERT_EQ(data.type().extent(), 3);
  868. ASSERT_EQ(data.get(nullptr, 0).cast<int>(), 3);
  869. ASSERT_EQ(data.get(nullptr, 1).cast<int>(), 5);
  870. ASSERT_EQ(data.get(nullptr, 2).cast<int>(), 7);
  871. ASSERT_FALSE(data.set(nullptr, 0, 'c'));
  872. ASSERT_EQ(data.get(nullptr, 0).cast<int>(), 3);
  873. ASSERT_TRUE(data.set(nullptr, 0, data.get(nullptr, 0).cast<int>()+2));
  874. ASSERT_TRUE(data.set(nullptr, 1, data.get(nullptr, 1).cast<int>()+2));
  875. ASSERT_TRUE(data.set(nullptr, 2, data.get(nullptr, 2).cast<int>()+2));
  876. ASSERT_EQ(data.get(nullptr, 0).cast<int>(), 5);
  877. ASSERT_EQ(data.get(nullptr, 1).cast<int>(), 7);
  878. ASSERT_EQ(data.get(nullptr, 2).cast<int>(), 9);
  879. }
  880. TEST_F(Meta, MetaDataArray) {
  881. auto data = entt::resolve<array_type>().data("local"_hs);
  882. array_type instance;
  883. instance.local[0] = 3;
  884. instance.local[1] = 5;
  885. instance.local[2] = 7;
  886. ASSERT_TRUE(data);
  887. ASSERT_NE(data, entt::meta_data{});
  888. ASSERT_EQ(data.parent(), entt::resolve("array"_hs));
  889. ASSERT_EQ(data.type(), entt::resolve<int[3]>());
  890. ASSERT_EQ(data.identifier(), "local"_hs);
  891. ASSERT_FALSE(data.is_const());
  892. ASSERT_FALSE(data.is_static());
  893. ASSERT_TRUE(data.type().is_array());
  894. ASSERT_EQ(data.type().extent(), 3);
  895. ASSERT_EQ(data.get(instance, 0).cast<int>(), 3);
  896. ASSERT_EQ(data.get(instance, 1).cast<int>(), 5);
  897. ASSERT_EQ(data.get(instance, 2).cast<int>(), 7);
  898. ASSERT_FALSE(data.set(instance, 0, 'c'));
  899. ASSERT_EQ(data.get(instance, 0).cast<int>(), 3);
  900. ASSERT_TRUE(data.set(instance, 0, data.get(instance, 0).cast<int>()+2));
  901. ASSERT_TRUE(data.set(instance, 1, data.get(instance, 1).cast<int>()+2));
  902. ASSERT_TRUE(data.set(instance, 2, data.get(instance, 2).cast<int>()+2));
  903. ASSERT_EQ(data.get(instance, 0).cast<int>(), 5);
  904. ASSERT_EQ(data.get(instance, 1).cast<int>(), 7);
  905. ASSERT_EQ(data.get(instance, 2).cast<int>(), 9);
  906. }
  907. TEST_F(Meta, MetaFunc) {
  908. auto func = entt::resolve<func_type>().func("f2"_hs);
  909. func_type instance{};
  910. ASSERT_TRUE(func);
  911. ASSERT_NE(func, entt::meta_func{});
  912. ASSERT_EQ(func.parent(), entt::resolve("func"_hs));
  913. ASSERT_EQ(func.identifier(), "f2"_hs);
  914. ASSERT_EQ(func.size(), entt::meta_func::size_type{2});
  915. ASSERT_FALSE(func.is_const());
  916. ASSERT_FALSE(func.is_static());
  917. ASSERT_EQ(func.ret(), entt::resolve<int>());
  918. ASSERT_EQ(func.arg(entt::meta_func::size_type{0}), entt::resolve<int>());
  919. ASSERT_EQ(func.arg(entt::meta_func::size_type{1}), entt::resolve<int>());
  920. ASSERT_FALSE(func.arg(entt::meta_func::size_type{2}));
  921. auto any = func.invoke(instance, 3, 2);
  922. auto empty = func.invoke(instance);
  923. ASSERT_FALSE(empty);
  924. ASSERT_TRUE(any);
  925. ASSERT_EQ(any.type(), entt::resolve<int>());
  926. ASSERT_EQ(any.cast<int>(), 4);
  927. ASSERT_EQ(func_type::value, 3);
  928. func.prop([](auto prop) {
  929. ASSERT_TRUE(prop);
  930. ASSERT_EQ(prop.key(), properties::prop_bool);
  931. ASSERT_FALSE(prop.value().template cast<bool>());
  932. });
  933. ASSERT_FALSE(func.prop(properties::prop_int));
  934. auto prop = func.prop(properties::prop_bool);
  935. ASSERT_TRUE(prop);
  936. ASSERT_EQ(prop.key(), properties::prop_bool);
  937. ASSERT_FALSE(prop.value().cast<bool>());
  938. }
  939. TEST_F(Meta, MetaFuncConst) {
  940. auto func = entt::resolve<func_type>().func("f1"_hs);
  941. func_type instance{};
  942. ASSERT_TRUE(func);
  943. ASSERT_EQ(func.parent(), entt::resolve("func"_hs));
  944. ASSERT_EQ(func.identifier(), "f1"_hs);
  945. ASSERT_EQ(func.size(), entt::meta_func::size_type{1});
  946. ASSERT_TRUE(func.is_const());
  947. ASSERT_FALSE(func.is_static());
  948. ASSERT_EQ(func.ret(), entt::resolve<int>());
  949. ASSERT_EQ(func.arg(entt::meta_func::size_type{0}), entt::resolve<int>());
  950. ASSERT_FALSE(func.arg(entt::meta_func::size_type{1}));
  951. auto any = func.invoke(instance, 4);
  952. auto empty = func.invoke(instance, 'c');
  953. ASSERT_FALSE(empty);
  954. ASSERT_TRUE(any);
  955. ASSERT_EQ(any.type(), entt::resolve<int>());
  956. ASSERT_EQ(any.cast<int>(), 16);
  957. func.prop([](auto prop) {
  958. ASSERT_TRUE(prop);
  959. ASSERT_EQ(prop.key(), properties::prop_bool);
  960. ASSERT_FALSE(prop.value().template cast<bool>());
  961. });
  962. ASSERT_FALSE(func.prop(properties::prop_int));
  963. auto prop = func.prop(properties::prop_bool);
  964. ASSERT_TRUE(prop);
  965. ASSERT_EQ(prop.key(), properties::prop_bool);
  966. ASSERT_FALSE(prop.value().cast<bool>());
  967. }
  968. TEST_F(Meta, MetaFuncRetVoid) {
  969. auto func = entt::resolve<func_type>().func("g"_hs);
  970. func_type instance{};
  971. ASSERT_TRUE(func);
  972. ASSERT_EQ(func.parent(), entt::resolve("func"_hs));
  973. ASSERT_EQ(func.identifier(), "g"_hs);
  974. ASSERT_EQ(func.size(), entt::meta_func::size_type{1});
  975. ASSERT_FALSE(func.is_const());
  976. ASSERT_FALSE(func.is_static());
  977. ASSERT_EQ(func.ret(), entt::resolve<void>());
  978. ASSERT_EQ(func.arg(entt::meta_func::size_type{0}), entt::resolve<int>());
  979. ASSERT_FALSE(func.arg(entt::meta_func::size_type{1}));
  980. auto any = func.invoke(instance, 5);
  981. ASSERT_FALSE(any);
  982. ASSERT_EQ(func_type::value, 25);
  983. func.prop([](auto prop) {
  984. ASSERT_TRUE(prop);
  985. ASSERT_EQ(prop.key(), properties::prop_bool);
  986. ASSERT_FALSE(prop.value().template cast<bool>());
  987. });
  988. ASSERT_FALSE(func.prop(properties::prop_int));
  989. auto prop = func.prop(properties::prop_bool);
  990. ASSERT_TRUE(prop);
  991. ASSERT_EQ(prop.key(), properties::prop_bool);
  992. ASSERT_FALSE(prop.value().cast<bool>());
  993. }
  994. TEST_F(Meta, MetaFuncStatic) {
  995. auto func = entt::resolve<func_type>().func("h"_hs);
  996. ASSERT_TRUE(func);
  997. ASSERT_EQ(func.parent(), entt::resolve("func"_hs));
  998. ASSERT_EQ(func.identifier(), "h"_hs);
  999. ASSERT_EQ(func.size(), entt::meta_func::size_type{1});
  1000. ASSERT_FALSE(func.is_const());
  1001. ASSERT_TRUE(func.is_static());
  1002. ASSERT_EQ(func.ret(), entt::resolve<int>());
  1003. ASSERT_EQ(func.arg(entt::meta_func::size_type{0}), entt::resolve<int>());
  1004. ASSERT_FALSE(func.arg(entt::meta_func::size_type{1}));
  1005. auto any = func.invoke({}, 42);
  1006. auto empty = func.invoke({}, 'c');
  1007. ASSERT_FALSE(empty);
  1008. ASSERT_TRUE(any);
  1009. ASSERT_EQ(any.type(), entt::resolve<int>());
  1010. ASSERT_EQ(any.cast<int>(), 42);
  1011. func.prop([](auto prop) {
  1012. ASSERT_TRUE(prop);
  1013. ASSERT_EQ(prop.key(), properties::prop_bool);
  1014. ASSERT_FALSE(prop.value().template cast<bool>());
  1015. });
  1016. ASSERT_FALSE(func.prop(properties::prop_int));
  1017. auto prop = func.prop(properties::prop_bool);
  1018. ASSERT_TRUE(prop);
  1019. ASSERT_EQ(prop.key(), properties::prop_bool);
  1020. ASSERT_FALSE(prop.value().cast<bool>());
  1021. }
  1022. TEST_F(Meta, MetaFuncStaticRetVoid) {
  1023. auto func = entt::resolve<func_type>().func("k"_hs);
  1024. ASSERT_TRUE(func);
  1025. ASSERT_EQ(func.parent(), entt::resolve("func"_hs));
  1026. ASSERT_EQ(func.identifier(), "k"_hs);
  1027. ASSERT_EQ(func.size(), entt::meta_func::size_type{1});
  1028. ASSERT_FALSE(func.is_const());
  1029. ASSERT_TRUE(func.is_static());
  1030. ASSERT_EQ(func.ret(), entt::resolve<void>());
  1031. ASSERT_EQ(func.arg(entt::meta_func::size_type{0}), entt::resolve<int>());
  1032. ASSERT_FALSE(func.arg(entt::meta_func::size_type{1}));
  1033. auto any = func.invoke({}, 42);
  1034. ASSERT_FALSE(any);
  1035. ASSERT_EQ(func_type::value, 42);
  1036. func.prop([](auto *prop) {
  1037. ASSERT_TRUE(prop);
  1038. ASSERT_EQ(prop->key(), properties::prop_bool);
  1039. ASSERT_FALSE(prop->value().template cast<bool>());
  1040. });
  1041. ASSERT_FALSE(func.prop(properties::prop_int));
  1042. auto prop = func.prop(properties::prop_bool);
  1043. ASSERT_TRUE(prop);
  1044. ASSERT_EQ(prop.key(), properties::prop_bool);
  1045. ASSERT_FALSE(prop.value().cast<bool>());
  1046. }
  1047. TEST_F(Meta, MetaFuncMetaAnyArgs) {
  1048. auto func = entt::resolve<func_type>().func("f1"_hs);
  1049. auto any = func.invoke(func_type{}, entt::meta_any{3});
  1050. ASSERT_TRUE(any);
  1051. ASSERT_EQ(any.type(), entt::resolve<int>());
  1052. ASSERT_EQ(any.cast<int>(), 9);
  1053. }
  1054. TEST_F(Meta, MetaFuncInvalidArgs) {
  1055. auto func = entt::resolve<func_type>().func("f1"_hs);
  1056. ASSERT_FALSE(func.invoke(empty_type{}, entt::meta_any{'c'}));
  1057. }
  1058. TEST_F(Meta, MetaFuncCastAndConvert) {
  1059. auto func = entt::resolve<func_type>().func("f3"_hs);
  1060. auto any = func.invoke(func_type{}, derived_type{}, 0, 3.);
  1061. ASSERT_TRUE(any);
  1062. ASSERT_EQ(any.type(), entt::resolve<int>());
  1063. ASSERT_EQ(any.cast<int>(), 9);
  1064. }
  1065. TEST_F(Meta, MetaType) {
  1066. auto type = entt::resolve<derived_type>();
  1067. ASSERT_TRUE(type);
  1068. ASSERT_NE(type, entt::meta_type{});
  1069. ASSERT_EQ(type.identifier(), "derived"_hs);
  1070. type.prop([](auto prop) {
  1071. ASSERT_TRUE(prop);
  1072. ASSERT_EQ(prop.key(), properties::prop_int);
  1073. ASSERT_EQ(prop.value(), 99);
  1074. });
  1075. ASSERT_FALSE(type.prop(properties::prop_bool));
  1076. auto prop = type.prop(properties::prop_int);
  1077. ASSERT_TRUE(prop);
  1078. ASSERT_EQ(prop.key(), properties::prop_int);
  1079. ASSERT_EQ(prop.value(), 99);
  1080. }
  1081. TEST_F(Meta, MetaTypeTraits) {
  1082. ASSERT_TRUE(entt::resolve<void>().is_void());
  1083. ASSERT_TRUE(entt::resolve<bool>().is_integral());
  1084. ASSERT_TRUE(entt::resolve<double>().is_floating_point());
  1085. ASSERT_TRUE(entt::resolve<properties>().is_enum());
  1086. ASSERT_TRUE(entt::resolve<union_type>().is_union());
  1087. ASSERT_TRUE(entt::resolve<derived_type>().is_class());
  1088. ASSERT_TRUE(entt::resolve<int *>().is_pointer());
  1089. ASSERT_TRUE(entt::resolve<decltype(empty_type::destroy)>().is_function());
  1090. ASSERT_TRUE(entt::resolve<decltype(&data_type::i)>().is_member_object_pointer());
  1091. ASSERT_TRUE(entt::resolve<decltype(&func_type::g)>().is_member_function_pointer());
  1092. }
  1093. TEST_F(Meta, MetaTypeRemovePointer) {
  1094. ASSERT_EQ(entt::resolve<void *>().remove_pointer(), entt::resolve<void>());
  1095. ASSERT_EQ(entt::resolve<int(*)(char, double)>().remove_pointer(), entt::resolve<int(char, double)>());
  1096. ASSERT_EQ(entt::resolve<derived_type>().remove_pointer(), entt::resolve<derived_type>());
  1097. }
  1098. TEST_F(Meta, MetaTypeBase) {
  1099. auto type = entt::resolve<derived_type>();
  1100. bool iterate = false;
  1101. type.base([&iterate](auto base) {
  1102. ASSERT_EQ(base.type(), entt::resolve<base_type>());
  1103. iterate = true;
  1104. });
  1105. ASSERT_TRUE(iterate);
  1106. ASSERT_EQ(type.base("base"_hs).type(), entt::resolve<base_type>());
  1107. }
  1108. TEST_F(Meta, MetaTypeConv) {
  1109. auto type = entt::resolve<double>();
  1110. bool iterate = false;
  1111. type.conv([&iterate](auto conv) {
  1112. ASSERT_EQ(conv.type(), entt::resolve<int>());
  1113. iterate = true;
  1114. });
  1115. ASSERT_TRUE(iterate);
  1116. auto conv = type.conv<int>();
  1117. ASSERT_EQ(conv.type(), entt::resolve<int>());
  1118. ASSERT_FALSE(type.conv<char>());
  1119. }
  1120. TEST_F(Meta, MetaTypeCtor) {
  1121. auto type = entt::resolve<derived_type>();
  1122. int counter{};
  1123. type.ctor([&counter](auto) {
  1124. ++counter;
  1125. });
  1126. ASSERT_EQ(counter, 2);
  1127. ASSERT_TRUE((type.ctor<const base_type &, int>()));
  1128. ASSERT_TRUE((type.ctor<const derived_type &, double>()));
  1129. }
  1130. TEST_F(Meta, MetaTypeDtor) {
  1131. ASSERT_TRUE(entt::resolve<fat_type>().dtor());
  1132. ASSERT_FALSE(entt::resolve<int>().dtor());
  1133. }
  1134. TEST_F(Meta, MetaTypeData) {
  1135. auto type = entt::resolve<data_type>();
  1136. int counter{};
  1137. type.data([&counter](auto) {
  1138. ++counter;
  1139. });
  1140. ASSERT_EQ(counter, 5);
  1141. ASSERT_TRUE(type.data("i"_hs));
  1142. }
  1143. TEST_F(Meta, MetaTypeFunc) {
  1144. auto type = entt::resolve<func_type>();
  1145. int counter{};
  1146. type.func([&counter](auto) {
  1147. ++counter;
  1148. });
  1149. ASSERT_EQ(counter, 6);
  1150. ASSERT_TRUE(type.func("f1"_hs));
  1151. }
  1152. TEST_F(Meta, MetaTypeConstruct) {
  1153. auto type = entt::resolve<derived_type>();
  1154. auto any = type.construct(base_type{}, 42, 'c');
  1155. ASSERT_TRUE(any);
  1156. ASSERT_TRUE(any.can_cast<derived_type>());
  1157. ASSERT_EQ(any.cast<derived_type>().i, 42);
  1158. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  1159. }
  1160. TEST_F(Meta, MetaTypeConstructMetaAnyArgs) {
  1161. auto type = entt::resolve<derived_type>();
  1162. auto any = type.construct(entt::meta_any{base_type{}}, entt::meta_any{42}, entt::meta_any{'c'});
  1163. ASSERT_TRUE(any);
  1164. ASSERT_TRUE(any.can_cast<derived_type>());
  1165. ASSERT_EQ(any.cast<derived_type>().i, 42);
  1166. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  1167. }
  1168. TEST_F(Meta, MetaTypeConstructInvalidArgs) {
  1169. auto type = entt::resolve<derived_type>();
  1170. auto any = type.construct(entt::meta_any{base_type{}}, entt::meta_any{'c'}, entt::meta_any{42});
  1171. ASSERT_FALSE(any);
  1172. }
  1173. TEST_F(Meta, MetaTypeLessArgs) {
  1174. auto type = entt::resolve<derived_type>();
  1175. auto any = type.construct(base_type{});
  1176. ASSERT_FALSE(any);
  1177. }
  1178. TEST_F(Meta, MetaTypeConstructCastAndConvert) {
  1179. auto type = entt::resolve<derived_type>();
  1180. auto any = type.construct(entt::meta_any{derived_type{}}, entt::meta_any{42.}, entt::meta_any{'c'});
  1181. ASSERT_TRUE(any);
  1182. ASSERT_TRUE(any.can_cast<derived_type>());
  1183. ASSERT_EQ(any.cast<derived_type>().i, 42);
  1184. ASSERT_EQ(any.cast<derived_type>().c, 'c');
  1185. }
  1186. TEST_F(Meta, MetaTypeDestroyDtor) {
  1187. auto type = entt::resolve<empty_type>();
  1188. ASSERT_EQ(empty_type::counter, 0);
  1189. ASSERT_TRUE(type.destroy(empty_type{}));
  1190. ASSERT_EQ(empty_type::counter, 1);
  1191. }
  1192. TEST_F(Meta, MetaTypeDestroyDtorInvalidArg) {
  1193. auto type = entt::resolve<empty_type>();
  1194. ASSERT_EQ(empty_type::counter, 0);
  1195. ASSERT_FALSE(type.destroy('c'));
  1196. ASSERT_EQ(empty_type::counter, 0);
  1197. }
  1198. TEST_F(Meta, MetaTypeDestroyDtorCastAndConvert) {
  1199. auto type = entt::resolve<empty_type>();
  1200. ASSERT_EQ(empty_type::counter, 0);
  1201. ASSERT_FALSE(type.destroy(fat_type{}));
  1202. ASSERT_EQ(empty_type::counter, 0);
  1203. ASSERT_FALSE(entt::resolve<int>().destroy(42.));
  1204. }
  1205. TEST_F(Meta, MetaTypeDestroyNoDtor) {
  1206. ASSERT_TRUE(entt::resolve<char>().destroy('c'));
  1207. }
  1208. TEST_F(Meta, MetaTypeDestroyNoDtorInvalidArg) {
  1209. ASSERT_FALSE(entt::resolve<char>().destroy(42));
  1210. }
  1211. TEST_F(Meta, MetaTypeDestroyNoDtorVoid) {
  1212. ASSERT_FALSE(entt::resolve<void>().destroy({}));
  1213. }
  1214. TEST_F(Meta, MetaTypeDestroyNoDtorCastAndConvert) {
  1215. ASSERT_FALSE(entt::resolve<int>().destroy(42.));
  1216. }
  1217. TEST_F(Meta, MetaDataFromBase) {
  1218. auto type = entt::resolve<concrete_type>();
  1219. concrete_type instance;
  1220. ASSERT_TRUE(type.data("i"_hs));
  1221. ASSERT_TRUE(type.data("j"_hs));
  1222. ASSERT_EQ(instance.i, 0);
  1223. ASSERT_EQ(instance.j, char{});
  1224. ASSERT_TRUE(type.data("i"_hs).set(instance, 3));
  1225. ASSERT_TRUE(type.data("j"_hs).set(instance, 'c'));
  1226. ASSERT_EQ(instance.i, 3);
  1227. ASSERT_EQ(instance.j, 'c');
  1228. }
  1229. TEST_F(Meta, MetaFuncFromBase) {
  1230. auto type = entt::resolve<concrete_type>();
  1231. auto base = entt::resolve<an_abstract_type>();
  1232. concrete_type instance;
  1233. ASSERT_TRUE(type.func("f"_hs));
  1234. ASSERT_TRUE(type.func("g"_hs));
  1235. ASSERT_TRUE(type.func("h"_hs));
  1236. ASSERT_EQ(type.func("f"_hs).parent(), entt::resolve<concrete_type>());
  1237. ASSERT_EQ(type.func("g"_hs).parent(), entt::resolve<an_abstract_type>());
  1238. ASSERT_EQ(type.func("h"_hs).parent(), entt::resolve<another_abstract_type>());
  1239. ASSERT_EQ(instance.i, 0);
  1240. ASSERT_EQ(instance.j, char{});
  1241. type.func("f"_hs).invoke(instance, 3);
  1242. type.func("h"_hs).invoke(instance, 'c');
  1243. ASSERT_EQ(instance.i, 9);
  1244. ASSERT_EQ(instance.j, 'c');
  1245. base.func("g"_hs).invoke(instance, 3);
  1246. ASSERT_EQ(instance.i, -3);
  1247. }
  1248. TEST_F(Meta, MetaPropFromBase) {
  1249. auto type = entt::resolve<concrete_type>();
  1250. auto prop_bool = type.prop(properties::prop_bool);
  1251. auto prop_int = type.prop(properties::prop_int);
  1252. ASSERT_TRUE(prop_bool);
  1253. ASSERT_TRUE(prop_int);
  1254. ASSERT_FALSE(prop_bool.value().cast<bool>());
  1255. ASSERT_EQ(prop_int.value().cast<int>(), 42);
  1256. }
  1257. TEST_F(Meta, AbstractClass) {
  1258. auto type = entt::resolve<an_abstract_type>();
  1259. concrete_type instance;
  1260. ASSERT_EQ(instance.i, 0);
  1261. type.func("f"_hs).invoke(instance, 3);
  1262. ASSERT_EQ(instance.i, 3);
  1263. type.func("g"_hs).invoke(instance, 3);
  1264. ASSERT_EQ(instance.i, -3);
  1265. }
  1266. TEST_F(Meta, EnumAndNamedConstants) {
  1267. auto type = entt::resolve<properties>();
  1268. ASSERT_TRUE(type.data("prop_bool"_hs));
  1269. ASSERT_TRUE(type.data("prop_int"_hs));
  1270. ASSERT_EQ(type.data("prop_bool"_hs).type(), type);
  1271. ASSERT_EQ(type.data("prop_int"_hs).type(), type);
  1272. ASSERT_FALSE(type.data("prop_bool"_hs).set({}, properties::prop_int));
  1273. ASSERT_FALSE(type.data("prop_int"_hs).set({}, properties::prop_bool));
  1274. ASSERT_EQ(type.data("prop_bool"_hs).get({}).cast<properties>(), properties::prop_bool);
  1275. ASSERT_EQ(type.data("prop_int"_hs).get({}).cast<properties>(), properties::prop_int);
  1276. }
  1277. TEST_F(Meta, ArithmeticTypeAndNamedConstants) {
  1278. auto type = entt::resolve<unsigned int>();
  1279. ASSERT_TRUE(type.data("min"_hs));
  1280. ASSERT_TRUE(type.data("max"_hs));
  1281. ASSERT_EQ(type.data("min"_hs).type(), type);
  1282. ASSERT_EQ(type.data("max"_hs).type(), type);
  1283. ASSERT_FALSE(type.data("min"_hs).set({}, 100u));
  1284. ASSERT_FALSE(type.data("max"_hs).set({}, 0u));
  1285. ASSERT_EQ(type.data("min"_hs).get({}).cast<unsigned int>(), 0u);
  1286. ASSERT_EQ(type.data("max"_hs).get({}).cast<unsigned int>(), 100u);
  1287. }
  1288. TEST_F(Meta, Unregister) {
  1289. ASSERT_FALSE(entt::unregister<float>());
  1290. ASSERT_TRUE(entt::unregister<double>());
  1291. ASSERT_TRUE(entt::unregister<char>());
  1292. ASSERT_TRUE(entt::unregister<properties>());
  1293. ASSERT_TRUE(entt::unregister<unsigned int>());
  1294. ASSERT_TRUE(entt::unregister<base_type>());
  1295. ASSERT_TRUE(entt::unregister<derived_type>());
  1296. ASSERT_TRUE(entt::unregister<empty_type>());
  1297. ASSERT_TRUE(entt::unregister<fat_type>());
  1298. ASSERT_TRUE(entt::unregister<data_type>());
  1299. ASSERT_TRUE(entt::unregister<func_type>());
  1300. ASSERT_TRUE(entt::unregister<setter_getter_type>());
  1301. ASSERT_TRUE(entt::unregister<an_abstract_type>());
  1302. ASSERT_TRUE(entt::unregister<another_abstract_type>());
  1303. ASSERT_TRUE(entt::unregister<concrete_type>());
  1304. ASSERT_FALSE(entt::unregister<double>());
  1305. ASSERT_FALSE(entt::resolve("char"_hs));
  1306. ASSERT_FALSE(entt::resolve("base"_hs));
  1307. ASSERT_FALSE(entt::resolve("derived"_hs));
  1308. ASSERT_FALSE(entt::resolve("empty"_hs));
  1309. ASSERT_FALSE(entt::resolve("fat"_hs));
  1310. ASSERT_FALSE(entt::resolve("data"_hs));
  1311. ASSERT_FALSE(entt::resolve("func"_hs));
  1312. ASSERT_FALSE(entt::resolve("setter_getter"_hs));
  1313. ASSERT_FALSE(entt::resolve("an_abstract_type"_hs));
  1314. ASSERT_FALSE(entt::resolve("another_abstract_type"_hs));
  1315. ASSERT_FALSE(entt::resolve("concrete"_hs));
  1316. Meta::SetUpAfterUnregistration();
  1317. entt::meta_any any{42.};
  1318. ASSERT_TRUE(any);
  1319. ASSERT_FALSE(any.can_convert<int>());
  1320. ASSERT_TRUE(any.can_convert<float>());
  1321. ASSERT_FALSE(entt::resolve("derived"_hs));
  1322. ASSERT_TRUE(entt::resolve("my_type"_hs));
  1323. entt::resolve<derived_type>().prop([](auto prop) {
  1324. ASSERT_TRUE(prop);
  1325. ASSERT_EQ(prop.key(), properties::prop_bool);
  1326. ASSERT_FALSE(prop.value().template cast<bool>());
  1327. });
  1328. ASSERT_FALSE((entt::resolve<derived_type>().ctor<const base_type &, int, char>()));
  1329. ASSERT_TRUE((entt::resolve<derived_type>().ctor<>()));
  1330. ASSERT_TRUE(entt::resolve("your_type"_hs).data("a_data_member"_hs));
  1331. ASSERT_FALSE(entt::resolve("your_type"_hs).data("another_data_member"_hs));
  1332. ASSERT_TRUE(entt::resolve("your_type"_hs).func("a_member_function"_hs));
  1333. ASSERT_FALSE(entt::resolve("your_type"_hs).func("another_member_function"_hs));
  1334. }