meta.cpp 50 KB

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