registry.cpp 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270
  1. #include <unordered_map>
  2. #include <unordered_set>
  3. #include <functional>
  4. #include <iterator>
  5. #include <memory>
  6. #include <cstdint>
  7. #include <type_traits>
  8. #include <gtest/gtest.h>
  9. #include <entt/entity/entt_traits.hpp>
  10. #include <entt/entity/registry.hpp>
  11. ENTT_NAMED_TYPE(int)
  12. struct listener {
  13. template<typename Component>
  14. void incr(entt::registry &registry, entt::entity entity) {
  15. ASSERT_TRUE(registry.valid(entity));
  16. ASSERT_TRUE(registry.has<Component>(entity));
  17. last = entity;
  18. ++counter;
  19. }
  20. template<typename Component>
  21. void decr(entt::registry &registry, entt::entity entity) {
  22. ASSERT_TRUE(registry.valid(entity));
  23. ASSERT_TRUE(registry.has<Component>(entity));
  24. last = entity;
  25. --counter;
  26. }
  27. entt::entity last;
  28. int counter{0};
  29. };
  30. TEST(Registry, Context) {
  31. entt::registry registry;
  32. ASSERT_EQ(registry.ctx<char>(), nullptr);
  33. ASSERT_EQ(registry.ctx<int>(), nullptr);
  34. ASSERT_EQ(registry.ctx<double>(), nullptr);
  35. registry.set<char>();
  36. registry.set<int>();
  37. registry.set<double>();
  38. ASSERT_NE(registry.ctx<char>(), nullptr);
  39. ASSERT_NE(registry.ctx<int>(), nullptr);
  40. ASSERT_NE(registry.ctx<double>(), nullptr);
  41. registry.unset<int>();
  42. registry.unset<double>();
  43. ASSERT_NE(registry.ctx<char>(), nullptr);
  44. ASSERT_EQ(registry.ctx<int>(), nullptr);
  45. ASSERT_EQ(registry.ctx<double>(), nullptr);
  46. registry.set<char>('c');
  47. registry.set<int>(42);
  48. registry.set<double>(1.);
  49. ASSERT_EQ(*registry.ctx<char>(), 'c');
  50. ASSERT_NE(registry.ctx<char>(), nullptr);
  51. ASSERT_EQ(registry.ctx<char>(), std::as_const(registry).ctx<char>());
  52. ASSERT_EQ(*registry.ctx<int>(), 42);
  53. ASSERT_NE(registry.ctx<int>(), nullptr);
  54. ASSERT_EQ(registry.ctx<int>(), std::as_const(registry).ctx<int>());
  55. ASSERT_EQ(*registry.ctx<double>(), 1.);
  56. ASSERT_NE(registry.ctx<double>(), nullptr);
  57. ASSERT_EQ(registry.ctx<double>(), std::as_const(registry).ctx<double>());
  58. ASSERT_EQ(registry.ctx<float>(), nullptr);
  59. }
  60. TEST(Registry, Types) {
  61. entt::registry registry;
  62. ASSERT_EQ(registry.type<int>(), registry.type<int>());
  63. ASSERT_NE(registry.type<double>(), registry.type<int>());
  64. }
  65. TEST(Registry, Functionalities) {
  66. entt::registry registry;
  67. ASSERT_EQ(registry.size(), entt::registry::size_type{0});
  68. ASSERT_EQ(registry.alive(), entt::registry::size_type{0});
  69. ASSERT_NO_THROW(registry.reserve(42));
  70. ASSERT_NO_THROW(registry.reserve<int>(8));
  71. ASSERT_NO_THROW(registry.reserve<char>(8));
  72. ASSERT_TRUE(registry.empty());
  73. ASSERT_EQ(registry.capacity(), entt::registry::size_type{42});
  74. ASSERT_EQ(registry.capacity<int>(), entt::registry::size_type{8});
  75. ASSERT_EQ(registry.capacity<char>(), entt::registry::size_type{8});
  76. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{0});
  77. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{0});
  78. ASSERT_TRUE(registry.empty<int>());
  79. ASSERT_TRUE(registry.empty<char>());
  80. const auto e0 = registry.create();
  81. const auto e1 = registry.create();
  82. registry.assign<int>(e1);
  83. registry.assign<char>(e1);
  84. ASSERT_TRUE(registry.has<>(e0));
  85. ASSERT_TRUE(registry.has<>(e1));
  86. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{1});
  87. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{1});
  88. ASSERT_FALSE(registry.empty<int>());
  89. ASSERT_FALSE(registry.empty<char>());
  90. ASSERT_NE(e0, e1);
  91. ASSERT_FALSE(registry.has<int>(e0));
  92. ASSERT_TRUE(registry.has<int>(e1));
  93. ASSERT_FALSE(registry.has<char>(e0));
  94. ASSERT_TRUE(registry.has<char>(e1));
  95. ASSERT_FALSE((registry.has<int, char>(e0)));
  96. ASSERT_TRUE((registry.has<int, char>(e1)));
  97. ASSERT_EQ(registry.try_get<int>(e0), nullptr);
  98. ASSERT_NE(registry.try_get<int>(e1), nullptr);
  99. ASSERT_EQ(registry.try_get<char>(e0), nullptr);
  100. ASSERT_NE(registry.try_get<char>(e1), nullptr);
  101. ASSERT_EQ(registry.try_get<double>(e0), nullptr);
  102. ASSERT_EQ(registry.try_get<double>(e1), nullptr);
  103. ASSERT_EQ(registry.assign<int>(e0, 42), 42);
  104. ASSERT_EQ(registry.assign<char>(e0, 'c'), 'c');
  105. ASSERT_NO_THROW(registry.remove<int>(e1));
  106. ASSERT_NO_THROW(registry.remove<char>(e1));
  107. ASSERT_TRUE(registry.has<int>(e0));
  108. ASSERT_FALSE(registry.has<int>(e1));
  109. ASSERT_TRUE(registry.has<char>(e0));
  110. ASSERT_FALSE(registry.has<char>(e1));
  111. ASSERT_TRUE((registry.has<int, char>(e0)));
  112. ASSERT_FALSE((registry.has<int, char>(e1)));
  113. const auto e2 = registry.create();
  114. registry.assign_or_replace<int>(e2, registry.get<int>(e0));
  115. registry.assign_or_replace<char>(e2, registry.get<char>(e0));
  116. ASSERT_TRUE(registry.has<int>(e2));
  117. ASSERT_TRUE(registry.has<char>(e2));
  118. ASSERT_EQ(registry.get<int>(e0), 42);
  119. ASSERT_EQ(registry.get<char>(e0), 'c');
  120. ASSERT_NE(registry.try_get<int>(e0), nullptr);
  121. ASSERT_NE(registry.try_get<char>(e0), nullptr);
  122. ASSERT_EQ(registry.try_get<double>(e0), nullptr);
  123. ASSERT_EQ(*registry.try_get<int>(e0), 42);
  124. ASSERT_EQ(*registry.try_get<char>(e0), 'c');
  125. ASSERT_EQ(std::get<0>(registry.get<int, char>(e0)), 42);
  126. ASSERT_EQ(*std::get<0>(registry.try_get<int, char, double>(e0)), 42);
  127. ASSERT_EQ(std::get<1>(static_cast<const entt::registry &>(registry).get<int, char>(e0)), 'c');
  128. ASSERT_EQ(*std::get<1>(static_cast<const entt::registry &>(registry).try_get<int, char, double>(e0)), 'c');
  129. ASSERT_EQ(registry.get<int>(e0), registry.get<int>(e2));
  130. ASSERT_EQ(registry.get<char>(e0), registry.get<char>(e2));
  131. ASSERT_NE(&registry.get<int>(e0), &registry.get<int>(e2));
  132. ASSERT_NE(&registry.get<char>(e0), &registry.get<char>(e2));
  133. ASSERT_NO_THROW(registry.replace<int>(e0, 0));
  134. ASSERT_EQ(registry.get<int>(e0), 0);
  135. ASSERT_NO_THROW(registry.assign_or_replace<int>(e0, 1));
  136. ASSERT_NO_THROW(registry.assign_or_replace<int>(e1, 1));
  137. ASSERT_EQ(static_cast<const entt::registry &>(registry).get<int>(e0), 1);
  138. ASSERT_EQ(static_cast<const entt::registry &>(registry).get<int>(e1), 1);
  139. ASSERT_EQ(registry.size(), entt::registry::size_type{3});
  140. ASSERT_EQ(registry.alive(), entt::registry::size_type{3});
  141. ASSERT_FALSE(registry.empty());
  142. ASSERT_EQ(registry.version(e2), entt::registry::version_type{0});
  143. ASSERT_EQ(registry.current(e2), entt::registry::version_type{0});
  144. ASSERT_NO_THROW(registry.destroy(e2));
  145. ASSERT_EQ(registry.version(e2), entt::registry::version_type{0});
  146. ASSERT_EQ(registry.current(e2), entt::registry::version_type{1});
  147. ASSERT_TRUE(registry.valid(e0));
  148. ASSERT_TRUE(registry.unsafe_valid(e0));
  149. ASSERT_TRUE(registry.valid(e1));
  150. ASSERT_TRUE(registry.unsafe_valid(e1));
  151. ASSERT_FALSE(registry.valid(e2));
  152. ASSERT_FALSE(registry.unsafe_valid(e2));
  153. ASSERT_EQ(registry.size(), entt::registry::size_type{3});
  154. ASSERT_EQ(registry.alive(), entt::registry::size_type{2});
  155. ASSERT_FALSE(registry.empty());
  156. ASSERT_NO_THROW(registry.reset());
  157. ASSERT_EQ(registry.size(), entt::registry::size_type{3});
  158. ASSERT_EQ(registry.alive(), entt::registry::size_type{0});
  159. ASSERT_TRUE(registry.empty());
  160. const auto e3 = registry.create();
  161. ASSERT_EQ(registry.get_or_assign<int>(e3, 3), 3);
  162. ASSERT_EQ(registry.get_or_assign<char>(e3, 'c'), 'c');
  163. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{1});
  164. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{1});
  165. ASSERT_FALSE(registry.empty<int>());
  166. ASSERT_FALSE(registry.empty<char>());
  167. ASSERT_TRUE(registry.has<int>(e3));
  168. ASSERT_TRUE(registry.has<char>(e3));
  169. ASSERT_EQ(registry.get<int>(e3), 3);
  170. ASSERT_EQ(registry.get<char>(e3), 'c');
  171. ASSERT_NO_THROW(registry.reset<int>());
  172. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{0});
  173. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{1});
  174. ASSERT_TRUE(registry.empty<int>());
  175. ASSERT_FALSE(registry.empty<char>());
  176. ASSERT_NO_THROW(registry.reset());
  177. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{0});
  178. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{0});
  179. ASSERT_TRUE(registry.empty<int>());
  180. ASSERT_TRUE(registry.empty<char>());
  181. const auto e4 = registry.create();
  182. const auto e5 = registry.create();
  183. registry.assign<int>(e4);
  184. ASSERT_NO_THROW(registry.reset<int>(e4));
  185. ASSERT_NO_THROW(registry.reset<int>(e5));
  186. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{0});
  187. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{0});
  188. ASSERT_TRUE(registry.empty<int>());
  189. }
  190. TEST(Registry, Identifiers) {
  191. entt::registry registry;
  192. const auto pre = registry.create();
  193. ASSERT_EQ(pre, registry.entity(pre));
  194. registry.destroy(pre);
  195. const auto post = registry.create();
  196. ASSERT_NE(pre, post);
  197. ASSERT_EQ(registry.entity(pre), registry.entity(post));
  198. ASSERT_NE(registry.version(pre), registry.version(post));
  199. ASSERT_NE(registry.version(pre), registry.current(pre));
  200. ASSERT_EQ(registry.version(post), registry.current(post));
  201. }
  202. TEST(Registry, RawData) {
  203. entt::registry registry;
  204. const auto entity = registry.create();
  205. ASSERT_EQ(registry.raw<int>(), nullptr);
  206. ASSERT_EQ(std::as_const(registry).raw<int>(), nullptr);
  207. ASSERT_EQ(std::as_const(registry).data<int>(), nullptr);
  208. registry.assign<int>(entity, 42);
  209. ASSERT_NE(registry.raw<int>(), nullptr);
  210. ASSERT_NE(std::as_const(registry).raw<int>(), nullptr);
  211. ASSERT_NE(std::as_const(registry).data<int>(), nullptr);
  212. ASSERT_EQ(*registry.raw<int>(), 42);
  213. ASSERT_EQ(*std::as_const(registry).raw<int>(), 42);
  214. ASSERT_EQ(*std::as_const(registry).data<int>(), entity);
  215. }
  216. TEST(Registry, CreateDestroyCornerCase) {
  217. entt::registry registry;
  218. const auto e0 = registry.create();
  219. const auto e1 = registry.create();
  220. registry.destroy(e0);
  221. registry.destroy(e1);
  222. registry.each([](auto) { FAIL(); });
  223. ASSERT_EQ(registry.current(e0), entt::registry::version_type{1});
  224. ASSERT_EQ(registry.current(e1), entt::registry::version_type{1});
  225. }
  226. TEST(Registry, VersionOverflow) {
  227. entt::registry registry;
  228. const auto entity = registry.create();
  229. registry.destroy(entity);
  230. ASSERT_EQ(registry.version(entity), entt::registry::version_type{});
  231. for(auto i = entt::entt_traits<entt::entity>::version_mask; i; --i) {
  232. ASSERT_NE(registry.current(entity), registry.version(entity));
  233. registry.destroy(registry.create());
  234. }
  235. ASSERT_EQ(registry.current(entity), registry.version(entity));
  236. }
  237. TEST(Registry, Each) {
  238. entt::registry registry;
  239. entt::registry::size_type tot;
  240. entt::registry::size_type match;
  241. registry.create();
  242. registry.assign<int>(registry.create());
  243. registry.create();
  244. registry.assign<int>(registry.create());
  245. registry.create();
  246. tot = 0u;
  247. match = 0u;
  248. registry.each([&](auto entity) {
  249. if(registry.has<int>(entity)) { ++match; }
  250. registry.create();
  251. ++tot;
  252. });
  253. ASSERT_EQ(tot, 5u);
  254. ASSERT_EQ(match, 2u);
  255. tot = 0u;
  256. match = 0u;
  257. registry.each([&](auto entity) {
  258. if(registry.has<int>(entity)) {
  259. registry.destroy(entity);
  260. ++match;
  261. }
  262. ++tot;
  263. });
  264. ASSERT_EQ(tot, 10u);
  265. ASSERT_EQ(match, 2u);
  266. tot = 0u;
  267. match = 0u;
  268. registry.each([&](auto entity) {
  269. if(registry.has<int>(entity)) { ++match; }
  270. registry.destroy(entity);
  271. ++tot;
  272. });
  273. ASSERT_EQ(tot, 8u);
  274. ASSERT_EQ(match, 0u);
  275. registry.each([&](auto) { FAIL(); });
  276. }
  277. TEST(Registry, Orphans) {
  278. entt::registry registry;
  279. entt::registry::size_type tot{};
  280. registry.assign<int>(registry.create());
  281. registry.create();
  282. registry.assign<int>(registry.create());
  283. registry.orphans([&](auto) { ++tot; });
  284. ASSERT_EQ(tot, 1u);
  285. tot = {};
  286. registry.each([&](auto entity) { registry.reset<int>(entity); });
  287. registry.orphans([&](auto) { ++tot; });
  288. ASSERT_EQ(tot, 3u);
  289. registry.reset();
  290. tot = {};
  291. registry.orphans([&](auto) { ++tot; });
  292. ASSERT_EQ(tot, 0u);
  293. }
  294. TEST(Registry, CreateDestroyEntities) {
  295. entt::registry registry;
  296. entt::entity pre{}, post{};
  297. for(int i = 0; i < 10; ++i) {
  298. const auto entity = registry.create();
  299. registry.assign<double>(entity);
  300. }
  301. registry.reset();
  302. for(int i = 0; i < 7; ++i) {
  303. const auto entity = registry.create();
  304. registry.assign<int>(entity);
  305. if(i == 3) { pre = entity; }
  306. }
  307. registry.reset();
  308. for(int i = 0; i < 5; ++i) {
  309. const auto entity = registry.create();
  310. if(i == 3) { post = entity; }
  311. }
  312. ASSERT_FALSE(registry.valid(pre));
  313. ASSERT_TRUE(registry.valid(post));
  314. ASSERT_NE(registry.version(pre), registry.version(post));
  315. ASSERT_EQ(registry.version(pre) + 1, registry.version(post));
  316. ASSERT_EQ(registry.current(pre), registry.current(post));
  317. }
  318. TEST(Registry, View) {
  319. entt::registry registry;
  320. auto mview = registry.view<int, char>();
  321. auto iview = registry.view<int>();
  322. auto cview = registry.view<char>();
  323. const auto e0 = registry.create();
  324. registry.assign<int>(e0, 0);
  325. registry.assign<char>(e0, 'c');
  326. const auto e1 = registry.create();
  327. registry.assign<int>(e1, 0);
  328. const auto e2 = registry.create();
  329. registry.assign<int>(e2, 0);
  330. registry.assign<char>(e2, 'c');
  331. ASSERT_EQ(iview.size(), decltype(iview)::size_type{3});
  332. ASSERT_EQ(cview.size(), decltype(cview)::size_type{2});
  333. decltype(mview)::size_type cnt{0};
  334. mview.each([&cnt](auto...) { ++cnt; });
  335. ASSERT_EQ(cnt, decltype(mview)::size_type{2});
  336. }
  337. TEST(Registry, NonOwningGroupInitOnFirstUse) {
  338. entt::registry registry;
  339. const auto e0 = registry.create();
  340. registry.assign<int>(e0, 0);
  341. registry.assign<char>(e0, 'c');
  342. const auto e1 = registry.create();
  343. registry.assign<int>(e1, 0);
  344. const auto e2 = registry.create();
  345. registry.assign<int>(e2, 0);
  346. registry.assign<char>(e2, 'c');
  347. ASSERT_FALSE(registry.owned<int>());
  348. ASSERT_FALSE(registry.owned<char>());
  349. auto group = registry.group<>(entt::get<int, char>);
  350. decltype(group)::size_type cnt{0};
  351. group.each([&cnt](auto...) { ++cnt; });
  352. ASSERT_FALSE(registry.owned<int>());
  353. ASSERT_FALSE(registry.owned<char>());
  354. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  355. }
  356. TEST(Registry, NonOwningGroupInitOnAssign) {
  357. entt::registry registry;
  358. auto group = registry.group<>(entt::get<int, char>);
  359. const auto e0 = registry.create();
  360. registry.assign<int>(e0, 0);
  361. registry.assign<char>(e0, 'c');
  362. const auto e1 = registry.create();
  363. registry.assign<int>(e1, 0);
  364. const auto e2 = registry.create();
  365. registry.assign<int>(e2, 0);
  366. registry.assign<char>(e2, 'c');
  367. ASSERT_FALSE(registry.owned<int>());
  368. ASSERT_FALSE(registry.owned<char>());
  369. decltype(group)::size_type cnt{0};
  370. group.each([&cnt](auto...) { ++cnt; });
  371. ASSERT_FALSE(registry.owned<int>());
  372. ASSERT_FALSE(registry.owned<char>());
  373. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  374. }
  375. TEST(Registry, FullOwningGroupInitOnFirstUse) {
  376. entt::registry registry;
  377. const auto e0 = registry.create();
  378. registry.assign<int>(e0, 0);
  379. registry.assign<char>(e0, 'c');
  380. const auto e1 = registry.create();
  381. registry.assign<int>(e1, 0);
  382. const auto e2 = registry.create();
  383. registry.assign<int>(e2, 0);
  384. registry.assign<char>(e2, 'c');
  385. ASSERT_FALSE(registry.owned<int>());
  386. ASSERT_FALSE(registry.owned<char>());
  387. auto group = registry.group<int, char>();
  388. decltype(group)::size_type cnt{0};
  389. group.each([&cnt](auto...) { ++cnt; });
  390. ASSERT_TRUE(registry.owned<int>());
  391. ASSERT_TRUE(registry.owned<char>());
  392. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  393. }
  394. TEST(Registry, FullOwningGroupInitOnAssign) {
  395. entt::registry registry;
  396. auto group = registry.group<int, char>();
  397. const auto e0 = registry.create();
  398. registry.assign<int>(e0, 0);
  399. registry.assign<char>(e0, 'c');
  400. const auto e1 = registry.create();
  401. registry.assign<int>(e1, 0);
  402. const auto e2 = registry.create();
  403. registry.assign<int>(e2, 0);
  404. registry.assign<char>(e2, 'c');
  405. ASSERT_TRUE(registry.owned<int>());
  406. ASSERT_TRUE(registry.owned<char>());
  407. decltype(group)::size_type cnt{0};
  408. group.each([&cnt](auto...) { ++cnt; });
  409. ASSERT_TRUE(registry.owned<int>());
  410. ASSERT_TRUE(registry.owned<char>());
  411. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  412. }
  413. TEST(Registry, PartialOwningGroupInitOnFirstUse) {
  414. entt::registry registry;
  415. const auto e0 = registry.create();
  416. registry.assign<int>(e0, 0);
  417. registry.assign<char>(e0, 'c');
  418. const auto e1 = registry.create();
  419. registry.assign<int>(e1, 1);
  420. const auto e2 = registry.create();
  421. registry.assign<int>(e2, 2);
  422. registry.assign<char>(e2, 'c');
  423. ASSERT_FALSE(registry.owned<int>());
  424. ASSERT_FALSE(registry.owned<char>());
  425. auto group = registry.group<int>(entt::get<char>);
  426. decltype(group)::size_type cnt{0};
  427. group.each([&cnt](auto...) { ++cnt; });
  428. ASSERT_TRUE(registry.owned<int>());
  429. ASSERT_FALSE(registry.owned<char>());
  430. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  431. }
  432. TEST(Registry, PartialOwningGroupInitOnAssign) {
  433. entt::registry registry;
  434. auto group = registry.group<int>(entt::get<char>);
  435. const auto e0 = registry.create();
  436. registry.assign<int>(e0, 0);
  437. registry.assign<char>(e0, 'c');
  438. const auto e1 = registry.create();
  439. registry.assign<int>(e1, 0);
  440. const auto e2 = registry.create();
  441. registry.assign<int>(e2, 0);
  442. registry.assign<char>(e2, 'c');
  443. ASSERT_TRUE(registry.owned<int>());
  444. ASSERT_FALSE(registry.owned<char>());
  445. decltype(group)::size_type cnt{0};
  446. group.each([&cnt](auto...) { ++cnt; });
  447. ASSERT_TRUE(registry.owned<int>());
  448. ASSERT_FALSE(registry.owned<char>());
  449. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  450. }
  451. TEST(Registry, CleanViewAfterReset) {
  452. entt::registry registry;
  453. auto view = registry.view<int, char>();
  454. const auto entity = registry.create();
  455. registry.assign<int>(entity, 0);
  456. registry.assign<char>(entity, 'c');
  457. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  458. registry.reset<char>(entity);
  459. ASSERT_EQ(view.size(), entt::registry::size_type{0});
  460. registry.assign<char>(entity, 'c');
  461. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  462. registry.reset<int>();
  463. ASSERT_EQ(view.size(), entt::registry::size_type{0});
  464. registry.assign<int>(entity, 0);
  465. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  466. registry.reset();
  467. ASSERT_EQ(view.size(), entt::registry::size_type{0});
  468. }
  469. TEST(Registry, CleanNonOwningGroupViewAfterReset) {
  470. entt::registry registry;
  471. auto group = registry.group<>(entt::get<int, char>);
  472. const auto entity = registry.create();
  473. registry.assign<int>(entity, 0);
  474. registry.assign<char>(entity, 'c');
  475. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  476. registry.reset<char>(entity);
  477. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  478. registry.assign<char>(entity, 'c');
  479. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  480. registry.reset<int>();
  481. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  482. registry.assign<int>(entity, 0);
  483. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  484. registry.reset();
  485. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  486. }
  487. TEST(Registry, CleanFullOwningGroupViewAfterReset) {
  488. entt::registry registry;
  489. auto group = registry.group<int, char>();
  490. const auto entity = registry.create();
  491. registry.assign<int>(entity, 0);
  492. registry.assign<char>(entity, 'c');
  493. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  494. registry.reset<char>(entity);
  495. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  496. registry.assign<char>(entity, 'c');
  497. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  498. registry.reset<int>();
  499. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  500. registry.assign<int>(entity, 0);
  501. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  502. registry.reset();
  503. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  504. }
  505. TEST(Registry, CleanPartialOwningGroupViewAfterReset) {
  506. entt::registry registry;
  507. auto group = registry.group<int>(entt::get<char>);
  508. const auto entity = registry.create();
  509. registry.assign<int>(entity, 0);
  510. registry.assign<char>(entity, 'c');
  511. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  512. registry.reset<char>(entity);
  513. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  514. registry.assign<char>(entity, 'c');
  515. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  516. registry.reset<int>();
  517. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  518. registry.assign<int>(entity, 0);
  519. ASSERT_EQ(group.size(), entt::registry::size_type{1});
  520. registry.reset();
  521. ASSERT_EQ(group.size(), entt::registry::size_type{0});
  522. }
  523. TEST(Registry, SortSingle) {
  524. entt::registry registry;
  525. int val = 0;
  526. registry.assign<int>(registry.create(), val++);
  527. registry.assign<int>(registry.create(), val++);
  528. registry.assign<int>(registry.create(), val++);
  529. for(auto entity: registry.view<int>()) {
  530. ASSERT_EQ(registry.get<int>(entity), --val);
  531. }
  532. registry.sort<int>(std::less<int>{});
  533. for(auto entity: registry.view<int>()) {
  534. ASSERT_EQ(registry.get<int>(entity), val++);
  535. }
  536. }
  537. TEST(Registry, SortMulti) {
  538. entt::registry registry;
  539. unsigned int uval = 0u;
  540. int ival = 0;
  541. for(auto i = 0; i < 3; ++i) {
  542. const auto entity = registry.create();
  543. registry.assign<unsigned int>(entity, uval++);
  544. registry.assign<int>(entity, ival++);
  545. }
  546. for(auto entity: registry.view<unsigned int>()) {
  547. ASSERT_EQ(registry.get<unsigned int>(entity), --uval);
  548. }
  549. for(auto entity: registry.view<int>()) {
  550. ASSERT_EQ(registry.get<int>(entity), --ival);
  551. }
  552. registry.sort<unsigned int>(std::less<unsigned int>{});
  553. registry.sort<int, unsigned int>();
  554. for(auto entity: registry.view<unsigned int>()) {
  555. ASSERT_EQ(registry.get<unsigned int>(entity), uval++);
  556. }
  557. for(auto entity: registry.view<int>()) {
  558. ASSERT_EQ(registry.get<int>(entity), ival++);
  559. }
  560. }
  561. TEST(Registry, ComponentsWithTypesFromStandardTemplateLibrary) {
  562. // see #37 - the test shouldn't crash, that's all
  563. entt::registry registry;
  564. const auto entity = registry.create();
  565. registry.assign<std::unordered_set<int>>(entity).insert(42);
  566. registry.destroy(entity);
  567. }
  568. TEST(Registry, ConstructWithComponents) {
  569. // it should compile, that's all
  570. entt::registry registry;
  571. const auto value = 0;
  572. registry.assign<int>(registry.create(), value);
  573. }
  574. TEST(Registry, MergeTwoRegistries) {
  575. entt::registry src;
  576. entt::registry dst;
  577. std::unordered_map<entt::entity, entt::entity> ref;
  578. auto merge = [&ref, &dst](const auto &view) {
  579. view.each([&](auto entity, const auto &component) {
  580. if(ref.find(entity) == ref.cend()) {
  581. const auto other = dst.create();
  582. dst.template assign<std::decay_t<decltype(component)>>(other, component);
  583. ref.emplace(entity, other);
  584. } else {
  585. using component_type = std::decay_t<decltype(component)>;
  586. dst.template assign<component_type>(ref[entity], component);
  587. }
  588. });
  589. };
  590. auto e0 = src.create();
  591. src.assign<int>(e0);
  592. src.assign<float>(e0);
  593. src.assign<double>(e0);
  594. auto e1 = src.create();
  595. src.assign<char>(e1);
  596. src.assign<float>(e1);
  597. src.assign<int>(e1);
  598. auto e2 = dst.create();
  599. dst.assign<int>(e2);
  600. dst.assign<char>(e2);
  601. dst.assign<double>(e2);
  602. auto e3 = dst.create();
  603. dst.assign<float>(e3);
  604. dst.assign<int>(e3);
  605. auto eq = [](auto begin, auto end) { ASSERT_EQ(begin, end); };
  606. auto ne = [](auto begin, auto end) { ASSERT_NE(begin, end); };
  607. eq(dst.view<int, float, double>().begin(), dst.view<int, float, double>().end());
  608. eq(dst.view<char, float, int>().begin(), dst.view<char, float, int>().end());
  609. merge(src.view<int>());
  610. merge(src.view<char>());
  611. merge(src.view<double>());
  612. merge(src.view<float>());
  613. ne(dst.view<int, float, double>().begin(), dst.view<int, float, double>().end());
  614. ne(dst.view<char, float, int>().begin(), dst.view<char, float, int>().end());
  615. }
  616. TEST(Registry, Signals) {
  617. entt::registry registry;
  618. listener listener;
  619. registry.construction<int>().connect<&listener::incr<int>>(&listener);
  620. registry.destruction<int>().connect<&listener::decr<int>>(&listener);
  621. auto e0 = registry.create();
  622. auto e1 = registry.create();
  623. registry.assign<int>(e0);
  624. registry.assign<int>(e1);
  625. ASSERT_EQ(listener.counter, 2);
  626. ASSERT_EQ(listener.last, e1);
  627. registry.remove<int>(e0);
  628. ASSERT_EQ(listener.counter, 1);
  629. ASSERT_EQ(listener.last, e0);
  630. registry.destruction<int>().disconnect<&listener::decr<int>>(&listener);
  631. registry.remove<int>(e1);
  632. ASSERT_EQ(listener.counter, 1);
  633. ASSERT_EQ(listener.last, e0);
  634. registry.construction<int>().disconnect<&listener::incr<int>>(&listener);
  635. registry.assign<int>(e1);
  636. ASSERT_EQ(listener.counter, 1);
  637. ASSERT_EQ(listener.last, e0);
  638. registry.construction<int>().connect<&listener::incr<int>>(&listener);
  639. registry.destruction<int>().connect<&listener::decr<int>>(&listener);
  640. registry.assign<int>(e0);
  641. registry.reset<int>(e1);
  642. ASSERT_EQ(listener.counter, 1);
  643. ASSERT_EQ(listener.last, e1);
  644. registry.reset<int>();
  645. ASSERT_EQ(listener.counter, 0);
  646. ASSERT_EQ(listener.last, e0);
  647. registry.assign<int>(e0);
  648. registry.assign<int>(e1);
  649. registry.destroy(e1);
  650. ASSERT_EQ(listener.counter, 1);
  651. ASSERT_EQ(listener.last, e1);
  652. }
  653. TEST(Registry, DestroyByComponents) {
  654. entt::registry registry;
  655. const auto e0 = registry.create();
  656. const auto e1 = registry.create();
  657. const auto e2 = registry.create();
  658. registry.assign<int>(e0);
  659. registry.assign<char>(e0);
  660. registry.assign<double>(e0);
  661. registry.assign<int>(e1);
  662. registry.assign<char>(e1);
  663. registry.assign<int>(e2);
  664. ASSERT_TRUE(registry.valid(e0));
  665. ASSERT_TRUE(registry.valid(e1));
  666. ASSERT_TRUE(registry.valid(e2));
  667. {
  668. const auto view = registry.view<int, char, double>();
  669. registry.destroy(view.begin(), view.end());
  670. }
  671. ASSERT_FALSE(registry.valid(e0));
  672. ASSERT_TRUE(registry.valid(e1));
  673. ASSERT_TRUE(registry.valid(e2));
  674. {
  675. const auto view = registry.view<int, char>();
  676. registry.destroy(view.begin(), view.end());
  677. }
  678. ASSERT_FALSE(registry.valid(e0));
  679. ASSERT_FALSE(registry.valid(e1));
  680. ASSERT_TRUE(registry.valid(e2));
  681. {
  682. const auto view = registry.view<int>();
  683. registry.destroy(view.begin(), view.end());
  684. }
  685. ASSERT_FALSE(registry.valid(e0));
  686. ASSERT_FALSE(registry.valid(e1));
  687. ASSERT_FALSE(registry.valid(e2));
  688. }
  689. TEST(Registry, SignalsOnAccommodate) {
  690. entt::registry registry;
  691. const auto entity = registry.create();
  692. const auto group = registry.group<>(entt::get<int, char>);
  693. registry.assign<int>(entity);
  694. registry.assign_or_replace<char>(entity);
  695. ASSERT_FALSE((group.empty()));
  696. }
  697. TEST(Registry, CreateManyEntitiesAtOnce) {
  698. entt::registry registry;
  699. entt::entity entities[3];
  700. const auto entity = registry.create();
  701. registry.destroy(registry.create());
  702. registry.destroy(entity);
  703. registry.destroy(registry.create());
  704. registry.create(std::begin(entities), std::end(entities));
  705. ASSERT_TRUE(registry.valid(entities[0]));
  706. ASSERT_TRUE(registry.valid(entities[1]));
  707. ASSERT_TRUE(registry.valid(entities[2]));
  708. ASSERT_EQ(registry.entity(entities[0]), entt::entity{0});
  709. ASSERT_EQ(registry.version(entities[0]), entt::registry::version_type{2});
  710. ASSERT_EQ(registry.entity(entities[1]), entt::entity{1});
  711. ASSERT_EQ(registry.version(entities[1]), entt::registry::version_type{1});
  712. ASSERT_EQ(registry.entity(entities[2]), entt::entity{2});
  713. ASSERT_EQ(registry.version(entities[2]), entt::registry::version_type{0});
  714. }
  715. TEST(Registry, CreateAnEntityWithComponents) {
  716. entt::registry registry;
  717. const auto &[entity, ivalue, cvalue] = registry.create<int, char>();
  718. ASSERT_FALSE(registry.empty<int>());
  719. ASSERT_FALSE(registry.empty<char>());
  720. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{1});
  721. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{1});
  722. ASSERT_TRUE((registry.has<int, char>(entity)));
  723. ivalue = 42;
  724. cvalue = 'c';
  725. ASSERT_EQ(registry.get<int>(entity), 42);
  726. ASSERT_EQ(registry.get<char>(entity), 'c');
  727. }
  728. TEST(Registry, CreateManyEntitiesWithComponentsAtOnce) {
  729. entt::registry registry;
  730. entt::entity entities[3];
  731. const auto entity = registry.create();
  732. registry.destroy(registry.create());
  733. registry.destroy(entity);
  734. registry.destroy(registry.create());
  735. const auto [iptr, cptr] = registry.create<int, char>(std::begin(entities), std::end(entities));
  736. ASSERT_FALSE(registry.empty<int>());
  737. ASSERT_FALSE(registry.empty<char>());
  738. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{3});
  739. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{3});
  740. ASSERT_TRUE(registry.valid(entities[0]));
  741. ASSERT_TRUE(registry.valid(entities[1]));
  742. ASSERT_TRUE(registry.valid(entities[2]));
  743. ASSERT_EQ(registry.entity(entities[0]), entt::entity{0});
  744. ASSERT_EQ(registry.version(entities[0]), entt::registry::version_type{2});
  745. ASSERT_EQ(registry.entity(entities[1]), entt::entity{1});
  746. ASSERT_EQ(registry.version(entities[1]), entt::registry::version_type{1});
  747. ASSERT_EQ(registry.entity(entities[2]), entt::entity{2});
  748. ASSERT_EQ(registry.version(entities[2]), entt::registry::version_type{0});
  749. ASSERT_TRUE((registry.has<int, char>(entities[0])));
  750. ASSERT_TRUE((registry.has<int, char>(entities[1])));
  751. ASSERT_TRUE((registry.has<int, char>(entities[2])));
  752. for(auto i = 0; i < 3; ++i) {
  753. iptr[i] = i;
  754. cptr[i] = char('a'+i);
  755. }
  756. for(auto i = 0; i < 3; ++i) {
  757. ASSERT_EQ(registry.get<int>(entities[i]), i);
  758. ASSERT_EQ(registry.get<char>(entities[i]), char('a'+i));
  759. }
  760. }
  761. TEST(Registry, CreateManyEntitiesWithComponentsAtOnceWithListener) {
  762. entt::registry registry;
  763. entt::entity entities[3];
  764. listener listener;
  765. registry.construction<int>().connect<&listener::incr<int>>(&listener);
  766. registry.create<int, char>(std::begin(entities), std::end(entities));
  767. ASSERT_EQ(listener.counter, 3);
  768. }
  769. TEST(Registry, NonOwningGroupInterleaved) {
  770. entt::registry registry;
  771. typename entt::entity entity = entt::null;
  772. entity = registry.create();
  773. registry.assign<int>(entity);
  774. registry.assign<char>(entity);
  775. const auto group = registry.group<>(entt::get<int, char>);
  776. entity = registry.create();
  777. registry.assign<int>(entity);
  778. registry.assign<char>(entity);
  779. decltype(group)::size_type cnt{0};
  780. group.each([&cnt](auto...) { ++cnt; });
  781. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  782. }
  783. TEST(Registry, FullOwningGroupInterleaved) {
  784. entt::registry registry;
  785. typename entt::entity entity = entt::null;
  786. entity = registry.create();
  787. registry.assign<int>(entity);
  788. registry.assign<char>(entity);
  789. const auto group = registry.group<int, char>();
  790. entity = registry.create();
  791. registry.assign<int>(entity);
  792. registry.assign<char>(entity);
  793. decltype(group)::size_type cnt{0};
  794. group.each([&cnt](auto...) { ++cnt; });
  795. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  796. }
  797. TEST(Registry, PartialOwningGroupInterleaved) {
  798. entt::registry registry;
  799. typename entt::entity entity = entt::null;
  800. entity = registry.create();
  801. registry.assign<int>(entity);
  802. registry.assign<char>(entity);
  803. const auto group = registry.group<int>(entt::get<char>);
  804. entity = registry.create();
  805. registry.assign<int>(entity);
  806. registry.assign<char>(entity);
  807. decltype(group)::size_type cnt{0};
  808. group.each([&cnt](auto...) { ++cnt; });
  809. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  810. }
  811. TEST(Registry, NonOwningGroupSortInterleaved) {
  812. entt::registry registry;
  813. const auto group = registry.group<>(entt::get<int, char>);
  814. const auto e0 = registry.create();
  815. registry.assign<int>(e0, 0);
  816. registry.assign<char>(e0, '0');
  817. const auto e1 = registry.create();
  818. registry.assign<int>(e1, 1);
  819. registry.assign<char>(e1, '1');
  820. registry.sort<int>([](auto lhs, auto rhs) { return lhs > rhs; });
  821. registry.sort<char>([](auto lhs, auto rhs) { return lhs < rhs; });
  822. const auto e2 = registry.create();
  823. registry.assign<int>(e2, 2);
  824. registry.assign<char>(e2, '2');
  825. group.each([e0, e1, e2](const auto entity, const auto &i, const auto &c) {
  826. if(entity == e0) {
  827. ASSERT_EQ(i, 0);
  828. ASSERT_EQ(c, '0');
  829. } else if(entity == e1) {
  830. ASSERT_EQ(i, 1);
  831. ASSERT_EQ(c, '1');
  832. } else if(entity == e2) {
  833. ASSERT_EQ(i, 2);
  834. ASSERT_EQ(c, '2');
  835. }
  836. });
  837. }
  838. TEST(Registry, Clone) {
  839. entt::registry registry;
  840. entt::registry other;
  841. registry.destroy(registry.create());
  842. const auto e0 = registry.create();
  843. registry.assign<int>(e0, 0);
  844. registry.assign<double>(e0, 0.0);
  845. const auto e1 = registry.create();
  846. registry.assign<int>(e1, 1);
  847. registry.assign<char>(e1, '1');
  848. registry.assign<double>(e1, 1.1);
  849. const auto e2 = registry.create();
  850. registry.assign<int>(e2, 2);
  851. registry.assign<char>(e2, '2');
  852. registry.destroy(e1);
  853. other = registry.clone<int, char>();
  854. ASSERT_EQ(other.size(), registry.size());
  855. ASSERT_EQ(other.alive(), registry.alive());
  856. ASSERT_TRUE(other.valid(e0));
  857. ASSERT_FALSE(other.valid(e1));
  858. ASSERT_TRUE(other.valid(e2));
  859. ASSERT_TRUE((other.has<int>(e0)));
  860. ASSERT_FALSE((other.has<double>(e0)));
  861. ASSERT_TRUE((other.has<int, char>(e2)));
  862. ASSERT_EQ(other.get<int>(e0), 0);
  863. ASSERT_EQ(other.get<int>(e2), 2);
  864. ASSERT_EQ(other.get<char>(e2), '2');
  865. other = registry.clone();
  866. ASSERT_EQ(other.size(), registry.size());
  867. ASSERT_EQ(other.alive(), registry.alive());
  868. ASSERT_TRUE(other.valid(e0));
  869. ASSERT_FALSE(other.valid(e1));
  870. ASSERT_TRUE(other.valid(e2));
  871. ASSERT_TRUE((other.has<int, double>(e0)));
  872. ASSERT_TRUE((other.has<int, char>(e2)));
  873. ASSERT_EQ(other.get<int>(e0), 0);
  874. ASSERT_EQ(other.get<double>(e0), 0.);
  875. ASSERT_EQ(other.get<int>(e2), 2);
  876. ASSERT_EQ(other.get<char>(e2), '2');
  877. other = registry.clone<char>();
  878. ASSERT_EQ(other.size(), registry.size());
  879. ASSERT_EQ(other.alive(), registry.alive());
  880. ASSERT_TRUE(other.valid(e0));
  881. ASSERT_FALSE(other.valid(e1));
  882. ASSERT_TRUE(other.valid(e2));
  883. ASSERT_FALSE((other.has<int>(e0)));
  884. ASSERT_FALSE((other.has<double>(e0)));
  885. ASSERT_FALSE((other.has<int>(e2)));
  886. ASSERT_TRUE((other.has<char>(e2)));
  887. ASSERT_TRUE(other.orphan(e0));
  888. ASSERT_EQ(other.get<char>(e2), '2');
  889. const auto entity = registry.create();
  890. listener listener;
  891. ASSERT_NE(e1, entity);
  892. ASSERT_EQ(registry.entity(e1), registry.entity(entity));
  893. registry.construction<char>().connect<&listener::incr<char>>(&listener);
  894. registry.destruction<char>().connect<&listener::decr<char>>(&listener);
  895. registry.assign<char>(entity, 'e');
  896. registry.assign<char>(e0, '0');
  897. registry.remove<char>(e0);
  898. ASSERT_EQ(listener.counter, 1);
  899. ASSERT_EQ(listener.last, e0);
  900. }
  901. TEST(Registry, GetOrAssign) {
  902. entt::registry registry;
  903. const auto entity = registry.create();
  904. const auto value = registry.get_or_assign<int>(entity, 3);
  905. ASSERT_TRUE(registry.has<int>(entity));
  906. ASSERT_EQ(registry.get<int>(entity), value);
  907. ASSERT_EQ(registry.get<int>(entity), 3);
  908. }
  909. TEST(Registry, Constness) {
  910. entt::registry registry;
  911. ASSERT_TRUE((std::is_same_v<decltype(registry.get<int>({})), int &>));
  912. ASSERT_TRUE((std::is_same_v<decltype(registry.get<int, char>({})), std::tuple<int &, char &>>));
  913. ASSERT_TRUE((std::is_same_v<decltype(registry.try_get<int>({})), int *>));
  914. ASSERT_TRUE((std::is_same_v<decltype(registry.try_get<int, char>({})), std::tuple<int *, char *>>));
  915. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).get<int>({})), const int &>));
  916. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).get<int, char>({})), std::tuple<const int &, const char &>>));
  917. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).try_get<int>({})), const int *>));
  918. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).try_get<int, char>({})), std::tuple<const int *, const char *>>));
  919. }
  920. TEST(Registry, BatchCreateAmbiguousCall) {
  921. struct ambiguous { std::uint32_t foo; std::uint64_t bar; };
  922. entt::registry registry;
  923. const auto entity = registry.create();
  924. std::uint32_t foo = 32u;
  925. std::uint64_t bar = 64u;
  926. // this should work, no other tests required
  927. registry.assign<ambiguous>(entity, foo, bar);
  928. }