registry.cpp 73 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359
  1. #include <cstddef>
  2. #include <functional>
  3. #include <iterator>
  4. #include <memory>
  5. #include <tuple>
  6. #include <type_traits>
  7. #include <unordered_set>
  8. #include <utility>
  9. #include <gtest/gtest.h>
  10. #include <entt/core/hashed_string.hpp>
  11. #include <entt/core/type_info.hpp>
  12. #include <entt/entity/entity.hpp>
  13. #include <entt/entity/registry.hpp>
  14. #include "../common/config.h"
  15. struct empty_type {};
  16. struct no_eto_type {
  17. static constexpr std::size_t page_size = 1024u;
  18. };
  19. bool operator==(const no_eto_type &lhs, const no_eto_type &rhs) {
  20. return &lhs == &rhs;
  21. }
  22. struct stable_type {
  23. static constexpr auto in_place_delete = true;
  24. int value;
  25. };
  26. struct non_default_constructible {
  27. non_default_constructible(int v)
  28. : value{v} {}
  29. int value;
  30. };
  31. struct aggregate {
  32. int value{};
  33. };
  34. struct listener {
  35. template<typename Type>
  36. static void sort(entt::registry &registry) {
  37. registry.sort<Type>([](auto lhs, auto rhs) { return lhs < rhs; });
  38. }
  39. void incr(const entt::registry &, entt::entity entity) {
  40. last = entity;
  41. ++counter;
  42. }
  43. void decr(const entt::registry &, entt::entity entity) {
  44. last = entity;
  45. --counter;
  46. }
  47. entt::entity last{entt::null};
  48. int counter{0};
  49. };
  50. struct owner {
  51. void receive(const entt::registry &ref) {
  52. parent = &ref;
  53. }
  54. const entt::registry *parent{nullptr};
  55. };
  56. struct destruction_order {
  57. using ctx_check_type = int;
  58. destruction_order(const entt::registry &ref, bool &ctx)
  59. : registry{&ref},
  60. ctx_check{&ctx} {
  61. *ctx_check = (registry->ctx().find<ctx_check_type>() != nullptr);
  62. }
  63. ~destruction_order() {
  64. *ctx_check = *ctx_check && (registry->ctx().find<ctx_check_type>() != nullptr);
  65. }
  66. private:
  67. const entt::registry *registry;
  68. bool *ctx_check{};
  69. };
  70. enum class small_entity : std::uint32_t {};
  71. struct small_entity_traits {
  72. using value_type = small_entity;
  73. using entity_type = uint32_t;
  74. using version_type = uint16_t;
  75. static constexpr entity_type entity_mask = 0xFF;
  76. static constexpr entity_type version_mask = 0x00;
  77. };
  78. template<>
  79. struct entt::entt_traits<small_entity>: entt::basic_entt_traits<small_entity_traits> {
  80. using base_type = entt::basic_entt_traits<small_entity_traits>;
  81. static constexpr auto page_size = ENTT_SPARSE_PAGE;
  82. };
  83. TEST(Registry, Context) {
  84. entt::registry registry;
  85. auto &ctx = registry.ctx();
  86. const auto &cctx = std::as_const(registry).ctx();
  87. ASSERT_FALSE(ctx.contains<char>());
  88. ASSERT_FALSE(cctx.contains<const int>());
  89. ASSERT_EQ(ctx.find<char>(), nullptr);
  90. ASSERT_EQ(cctx.find<const int>(), nullptr);
  91. ctx.emplace<char>();
  92. ctx.emplace<int>();
  93. ASSERT_TRUE(ctx.contains<char>());
  94. ASSERT_TRUE(cctx.contains<int>());
  95. ASSERT_NE(ctx.find<const char>(), nullptr);
  96. ASSERT_NE(cctx.find<const int>(), nullptr);
  97. ASSERT_FALSE(ctx.erase<double>());
  98. ASSERT_TRUE(ctx.erase<int>());
  99. ASSERT_TRUE(ctx.contains<const char>());
  100. ASSERT_FALSE(cctx.contains<const int>());
  101. ASSERT_NE(ctx.find<char>(), nullptr);
  102. ASSERT_EQ(cctx.find<int>(), nullptr);
  103. ASSERT_FALSE(ctx.erase<int>());
  104. ASSERT_TRUE(ctx.erase<char>());
  105. ctx.emplace<char>('c');
  106. ctx.emplace<int>(42);
  107. ASSERT_EQ(ctx.emplace<char>('a'), 'c');
  108. ASSERT_EQ(ctx.find<const char>(), cctx.find<char>());
  109. ASSERT_EQ(ctx.get<char>(), cctx.get<const char>());
  110. ASSERT_EQ(ctx.get<char>(), 'c');
  111. ASSERT_EQ(ctx.emplace<const int>(0), 42);
  112. ASSERT_EQ(ctx.find<const int>(), cctx.find<int>());
  113. ASSERT_EQ(ctx.get<int>(), cctx.get<const int>());
  114. ASSERT_EQ(ctx.get<int>(), 42);
  115. ASSERT_EQ(ctx.find<double>(), nullptr);
  116. ASSERT_EQ(cctx.find<double>(), nullptr);
  117. ASSERT_EQ(ctx.insert_or_assign<char>('a'), 'a');
  118. ASSERT_EQ(ctx.find<const char>(), cctx.find<char>());
  119. ASSERT_EQ(ctx.get<char>(), cctx.get<const char>());
  120. ASSERT_EQ(ctx.get<const char>(), 'a');
  121. ASSERT_EQ(ctx.insert_or_assign<const int>(0), 0);
  122. ASSERT_EQ(ctx.find<const int>(), cctx.find<int>());
  123. ASSERT_EQ(ctx.get<int>(), cctx.get<const int>());
  124. ASSERT_EQ(ctx.get<int>(), 0);
  125. }
  126. TEST(Registry, ContextHint) {
  127. using namespace entt::literals;
  128. entt::registry registry;
  129. auto &ctx = registry.ctx();
  130. const auto &cctx = std::as_const(registry).ctx();
  131. ctx.emplace<int>(42);
  132. ctx.emplace_as<int>("other"_hs, 3);
  133. ASSERT_TRUE(ctx.contains<int>());
  134. ASSERT_TRUE(cctx.contains<const int>("other"_hs));
  135. ASSERT_FALSE(ctx.contains<char>("other"_hs));
  136. ASSERT_NE(cctx.find<const int>(), nullptr);
  137. ASSERT_NE(ctx.find<int>("other"_hs), nullptr);
  138. ASSERT_EQ(cctx.find<const char>("other"_hs), nullptr);
  139. ASSERT_EQ(ctx.get<int>(), 42);
  140. ASSERT_EQ(cctx.get<const int>("other"_hs), 3);
  141. ctx.insert_or_assign(3);
  142. ctx.insert_or_assign("other"_hs, 42);
  143. ASSERT_EQ(ctx.get<const int>(), 3);
  144. ASSERT_EQ(cctx.get<int>("other"_hs), 42);
  145. ASSERT_FALSE(ctx.erase<char>("other"_hs));
  146. ASSERT_TRUE(ctx.erase<int>());
  147. ASSERT_TRUE(cctx.contains<int>("other"_hs));
  148. ASSERT_EQ(ctx.get<int>("other"_hs), 42);
  149. ASSERT_TRUE(ctx.erase<int>("other"_hs));
  150. ASSERT_FALSE(cctx.contains<int>("other"_hs));
  151. ASSERT_EQ(ctx.find<int>("other"_hs), nullptr);
  152. }
  153. TEST(Registry, ContextAsRef) {
  154. entt::registry registry;
  155. int value{3};
  156. registry.ctx().emplace<int &>(value);
  157. ASSERT_NE(registry.ctx().find<int>(), nullptr);
  158. ASSERT_NE(registry.ctx().find<const int>(), nullptr);
  159. ASSERT_NE(std::as_const(registry).ctx().find<const int>(), nullptr);
  160. ASSERT_EQ(registry.ctx().get<const int>(), 3);
  161. ASSERT_EQ(registry.ctx().get<int>(), 3);
  162. registry.ctx().get<int>() = 42;
  163. ASSERT_EQ(registry.ctx().get<int>(), 42);
  164. ASSERT_EQ(value, 42);
  165. value = 3;
  166. ASSERT_EQ(std::as_const(registry).ctx().get<const int>(), 3);
  167. }
  168. TEST(Registry, ContextAsConstRef) {
  169. entt::registry registry;
  170. int value{3};
  171. registry.ctx().emplace<const int &>(value);
  172. ASSERT_EQ(registry.ctx().find<int>(), nullptr);
  173. ASSERT_NE(registry.ctx().find<const int>(), nullptr);
  174. ASSERT_NE(std::as_const(registry).ctx().find<const int>(), nullptr);
  175. ASSERT_EQ(registry.ctx().get<const int>(), 3);
  176. value = 42;
  177. ASSERT_EQ(std::as_const(registry).ctx().get<const int>(), 42);
  178. }
  179. TEST(Registry, Functionalities) {
  180. using traits_type = entt::entt_traits<entt::entity>;
  181. entt::registry registry;
  182. ASSERT_NO_FATAL_FAILURE([[maybe_unused]] auto alloc = registry.get_allocator());
  183. ASSERT_EQ(registry.storage<entt::entity>().size(), 0u);
  184. ASSERT_EQ(registry.alive(), 0u);
  185. ASSERT_NO_FATAL_FAILURE(registry.reserve(42));
  186. ASSERT_EQ(registry.capacity(), 42u);
  187. ASSERT_TRUE(registry.empty());
  188. ASSERT_EQ(registry.storage<int>().size(), 0u);
  189. ASSERT_EQ(registry.storage<char>().size(), 0u);
  190. ASSERT_TRUE(registry.storage<int>().empty());
  191. ASSERT_TRUE(registry.storage<char>().empty());
  192. const auto e0 = registry.create();
  193. const auto e1 = registry.create();
  194. registry.emplace<int>(e1);
  195. registry.emplace<char>(e1);
  196. ASSERT_TRUE(registry.all_of<>(e0));
  197. ASSERT_FALSE(registry.any_of<>(e1));
  198. ASSERT_EQ(registry.storage<int>().size(), 1u);
  199. ASSERT_EQ(registry.storage<char>().size(), 1u);
  200. ASSERT_FALSE(registry.storage<int>().empty());
  201. ASSERT_FALSE(registry.storage<char>().empty());
  202. ASSERT_NE(e0, e1);
  203. ASSERT_FALSE((registry.all_of<int, const char>(e0)));
  204. ASSERT_TRUE((registry.all_of<const int, char>(e1)));
  205. ASSERT_FALSE((registry.any_of<int, const double>(e0)));
  206. ASSERT_TRUE((registry.any_of<const int, double>(e1)));
  207. ASSERT_EQ(registry.try_get<int>(e0), nullptr);
  208. ASSERT_NE(registry.try_get<int>(e1), nullptr);
  209. ASSERT_EQ(registry.try_get<char>(e0), nullptr);
  210. ASSERT_NE(registry.try_get<char>(e1), nullptr);
  211. ASSERT_EQ(registry.try_get<double>(e0), nullptr);
  212. ASSERT_EQ(registry.try_get<double>(e1), nullptr);
  213. ASSERT_EQ(registry.emplace<int>(e0, 42), 42);
  214. ASSERT_EQ(registry.emplace<char>(e0, 'c'), 'c');
  215. ASSERT_NO_FATAL_FAILURE(registry.erase<int>(e1));
  216. ASSERT_NO_FATAL_FAILURE(registry.erase<char>(e1));
  217. ASSERT_TRUE((registry.all_of<const int, char>(e0)));
  218. ASSERT_FALSE((registry.all_of<int, const char>(e1)));
  219. ASSERT_TRUE((registry.any_of<const int, double>(e0)));
  220. ASSERT_FALSE((registry.any_of<int, const double>(e1)));
  221. const auto e2 = registry.create();
  222. registry.emplace_or_replace<int>(e2, registry.get<int>(e0));
  223. registry.emplace_or_replace<char>(e2, registry.get<char>(e0));
  224. ASSERT_TRUE((registry.all_of<int, char>(e2)));
  225. ASSERT_EQ(registry.get<int>(e0), 42);
  226. ASSERT_EQ(registry.get<char>(e0), 'c');
  227. ASSERT_NE(registry.try_get<int>(e0), nullptr);
  228. ASSERT_NE(registry.try_get<char>(e0), nullptr);
  229. ASSERT_EQ(registry.try_get<double>(e0), nullptr);
  230. ASSERT_EQ(*registry.try_get<int>(e0), 42);
  231. ASSERT_EQ(*registry.try_get<char>(e0), 'c');
  232. ASSERT_EQ(std::get<0>(registry.get<int, char>(e0)), 42);
  233. ASSERT_EQ(*std::get<0>(registry.try_get<int, char, double>(e0)), 42);
  234. ASSERT_EQ(std::get<1>(static_cast<const entt::registry &>(registry).get<int, char>(e0)), 'c');
  235. ASSERT_EQ(*std::get<1>(static_cast<const entt::registry &>(registry).try_get<int, char, double>(e0)), 'c');
  236. ASSERT_EQ(registry.get<int>(e0), registry.get<int>(e2));
  237. ASSERT_EQ(registry.get<char>(e0), registry.get<char>(e2));
  238. ASSERT_NE(&registry.get<int>(e0), &registry.get<int>(e2));
  239. ASSERT_NE(&registry.get<char>(e0), &registry.get<char>(e2));
  240. ASSERT_EQ(registry.patch<int>(e0, [](auto &instance) { instance = 2; }), 2);
  241. ASSERT_EQ(registry.replace<int>(e0, 3), 3);
  242. ASSERT_NO_FATAL_FAILURE(registry.emplace_or_replace<int>(e0, 1));
  243. ASSERT_NO_FATAL_FAILURE(registry.emplace_or_replace<int>(e1, 1));
  244. ASSERT_EQ(static_cast<const entt::registry &>(registry).get<int>(e0), 1);
  245. ASSERT_EQ(static_cast<const entt::registry &>(registry).get<int>(e1), 1);
  246. ASSERT_EQ(registry.storage<entt::entity>().size(), 3u);
  247. ASSERT_EQ(registry.alive(), 3u);
  248. ASSERT_FALSE(registry.empty());
  249. ASSERT_EQ(traits_type::to_version(e2), 0u);
  250. ASSERT_EQ(registry.current(e2), 0u);
  251. ASSERT_NO_FATAL_FAILURE(registry.destroy(e2));
  252. ASSERT_EQ(traits_type::to_version(e2), 0u);
  253. ASSERT_EQ(registry.current(e2), 1u);
  254. ASSERT_TRUE(registry.valid(e0));
  255. ASSERT_TRUE(registry.valid(e1));
  256. ASSERT_FALSE(registry.valid(e2));
  257. ASSERT_EQ(registry.storage<entt::entity>().size(), 3u);
  258. ASSERT_EQ(registry.alive(), 2u);
  259. ASSERT_FALSE(registry.empty());
  260. ASSERT_NO_FATAL_FAILURE(registry.clear());
  261. ASSERT_EQ(registry.storage<entt::entity>().size(), 3u);
  262. ASSERT_EQ(registry.alive(), 0u);
  263. ASSERT_TRUE(registry.empty());
  264. const auto e3 = registry.create();
  265. ASSERT_EQ(registry.get_or_emplace<int>(e3, 3), 3);
  266. ASSERT_EQ(registry.get_or_emplace<char>(e3, 'c'), 'c');
  267. ASSERT_EQ(registry.storage<int>().size(), 1u);
  268. ASSERT_EQ(registry.storage<char>().size(), 1u);
  269. ASSERT_FALSE(registry.storage<int>().empty());
  270. ASSERT_FALSE(registry.storage<char>().empty());
  271. ASSERT_TRUE((registry.all_of<int, char>(e3)));
  272. ASSERT_EQ(registry.get<int>(e3), 3);
  273. ASSERT_EQ(registry.get<char>(e3), 'c');
  274. ASSERT_NO_FATAL_FAILURE(registry.clear<int>());
  275. ASSERT_EQ(registry.storage<int>().size(), 0u);
  276. ASSERT_EQ(registry.storage<char>().size(), 1u);
  277. ASSERT_TRUE(registry.storage<int>().empty());
  278. ASSERT_FALSE(registry.storage<char>().empty());
  279. ASSERT_NO_FATAL_FAILURE(registry.clear());
  280. ASSERT_EQ(registry.storage<int>().size(), 0u);
  281. ASSERT_EQ(registry.storage<char>().size(), 0u);
  282. ASSERT_TRUE(registry.storage<int>().empty());
  283. ASSERT_TRUE(registry.storage<char>().empty());
  284. const auto e4 = registry.create();
  285. const auto e5 = registry.create();
  286. registry.emplace<int>(e4);
  287. ASSERT_EQ(registry.remove<int>(e4), 1u);
  288. ASSERT_EQ(registry.remove<int>(e5), 0u);
  289. ASSERT_EQ(registry.storage<int>().size(), 0u);
  290. ASSERT_EQ(registry.storage<char>().size(), 0u);
  291. ASSERT_TRUE(registry.storage<int>().empty());
  292. }
  293. TEST(Registry, Constructors) {
  294. entt::registry registry;
  295. entt::registry other{42};
  296. ASSERT_TRUE(registry.empty());
  297. ASSERT_TRUE(other.empty());
  298. ASSERT_EQ(registry.released(), 0u);
  299. ASSERT_EQ(other.released(), 0u);
  300. }
  301. TEST(Registry, Move) {
  302. entt::registry registry;
  303. const auto entity = registry.create();
  304. owner test{};
  305. registry.on_construct<int>().connect<&owner::receive>(test);
  306. registry.on_destroy<int>().connect<&owner::receive>(test);
  307. ASSERT_EQ(test.parent, nullptr);
  308. registry.emplace<int>(entity);
  309. ASSERT_EQ(test.parent, &registry);
  310. entt::registry other{std::move(registry)};
  311. other.erase<int>(entity);
  312. registry = {};
  313. registry.emplace<int>(registry.create(entity));
  314. ASSERT_EQ(test.parent, &other);
  315. registry = std::move(other);
  316. registry.emplace<int>(entity);
  317. registry.emplace<int>(registry.create(entity));
  318. ASSERT_EQ(test.parent, &registry);
  319. }
  320. TEST(Registry, Swap) {
  321. entt::registry registry;
  322. const auto entity = registry.create();
  323. owner test{};
  324. registry.on_construct<int>().connect<&owner::receive>(test);
  325. registry.on_destroy<int>().connect<&owner::receive>(test);
  326. ASSERT_EQ(test.parent, nullptr);
  327. registry.emplace<int>(entity);
  328. ASSERT_EQ(test.parent, &registry);
  329. entt::registry other;
  330. other.swap(registry);
  331. other.erase<int>(entity);
  332. registry = {};
  333. registry.emplace<int>(registry.create(entity));
  334. ASSERT_EQ(test.parent, &other);
  335. registry.swap(other);
  336. registry.emplace<int>(entity);
  337. registry.emplace<int>(registry.create(entity));
  338. ASSERT_EQ(test.parent, &registry);
  339. }
  340. TEST(Registry, ReplaceAggregate) {
  341. entt::registry registry;
  342. const auto entity = registry.create();
  343. registry.emplace<aggregate>(entity, 0);
  344. auto &instance = registry.replace<aggregate>(entity, 42);
  345. ASSERT_EQ(instance.value, 42);
  346. }
  347. TEST(Registry, EmplaceOrReplaceAggregate) {
  348. entt::registry registry;
  349. const auto entity = registry.create();
  350. auto &instance = registry.emplace_or_replace<aggregate>(entity, 42);
  351. ASSERT_EQ(instance.value, 42);
  352. }
  353. TEST(Registry, Identifiers) {
  354. using traits_type = entt::entt_traits<entt::entity>;
  355. entt::registry registry;
  356. const auto pre = registry.create();
  357. ASSERT_EQ(traits_type::to_integral(pre), traits_type::to_entity(pre));
  358. registry.release(pre);
  359. const auto post = registry.create();
  360. ASSERT_NE(pre, post);
  361. ASSERT_EQ(traits_type::to_entity(pre), traits_type::to_entity(post));
  362. ASSERT_NE(traits_type::to_version(pre), traits_type::to_version(post));
  363. ASSERT_NE(traits_type::to_version(pre), registry.current(pre));
  364. ASSERT_EQ(traits_type::to_version(post), registry.current(post));
  365. const auto invalid = traits_type::combine(traits_type::to_entity(post) + 1u, {});
  366. ASSERT_EQ(traits_type::to_version(invalid), typename traits_type::version_type{});
  367. ASSERT_EQ(registry.current(invalid), traits_type::to_version(entt::tombstone));
  368. }
  369. TEST(Registry, Data) {
  370. using traits_type = entt::entt_traits<entt::entity>;
  371. entt::registry registry;
  372. ASSERT_EQ(std::as_const(registry).data(), nullptr);
  373. const auto entity = registry.create();
  374. ASSERT_EQ(*std::as_const(registry).data(), entity);
  375. const auto other = registry.create();
  376. registry.release(entity);
  377. ASSERT_EQ(*std::as_const(registry).data(), other);
  378. ASSERT_EQ(*(std::as_const(registry).data() + 1u), traits_type::next(entity));
  379. }
  380. TEST(Registry, CreateManyEntitiesAtOnce) {
  381. using traits_type = entt::entt_traits<entt::entity>;
  382. entt::registry registry;
  383. entt::entity entities[3];
  384. const auto entity = registry.create();
  385. registry.release(registry.create());
  386. registry.release(entity);
  387. registry.release(registry.create());
  388. registry.create(std::begin(entities), std::end(entities));
  389. ASSERT_TRUE(registry.valid(entities[0]));
  390. ASSERT_TRUE(registry.valid(entities[1]));
  391. ASSERT_TRUE(registry.valid(entities[2]));
  392. ASSERT_EQ(traits_type::to_entity(entities[0]), 0u);
  393. ASSERT_EQ(traits_type::to_version(entities[0]), 2u);
  394. ASSERT_EQ(traits_type::to_entity(entities[1]), 1u);
  395. ASSERT_EQ(traits_type::to_version(entities[1]), 1u);
  396. ASSERT_EQ(traits_type::to_entity(entities[2]), 2u);
  397. ASSERT_EQ(traits_type::to_version(entities[2]), 0u);
  398. }
  399. TEST(Registry, CreateManyEntitiesAtOnceWithListener) {
  400. entt::registry registry;
  401. entt::entity entities[3];
  402. listener listener;
  403. registry.on_construct<int>().connect<&listener::incr>(listener);
  404. registry.create(std::begin(entities), std::end(entities));
  405. registry.insert(std::begin(entities), std::end(entities), 42);
  406. registry.insert(std::begin(entities), std::end(entities), 'c');
  407. ASSERT_EQ(registry.get<int>(entities[0]), 42);
  408. ASSERT_EQ(registry.get<char>(entities[1]), 'c');
  409. ASSERT_EQ(listener.counter, 3);
  410. registry.on_construct<int>().disconnect<&listener::incr>(listener);
  411. registry.on_construct<empty_type>().connect<&listener::incr>(listener);
  412. registry.create(std::begin(entities), std::end(entities));
  413. registry.insert(std::begin(entities), std::end(entities), 'a');
  414. registry.insert<empty_type>(std::begin(entities), std::end(entities));
  415. ASSERT_TRUE(registry.all_of<empty_type>(entities[0]));
  416. ASSERT_EQ(registry.get<char>(entities[2]), 'a');
  417. ASSERT_EQ(listener.counter, 6);
  418. }
  419. TEST(Registry, CreateWithHint) {
  420. using traits_type = entt::entt_traits<entt::entity>;
  421. entt::registry registry;
  422. auto e3 = registry.create(entt::entity{3});
  423. auto e2 = registry.create(entt::entity{3});
  424. ASSERT_EQ(e2, entt::entity{1});
  425. ASSERT_TRUE(registry.valid(entt::entity{0}));
  426. ASSERT_TRUE(registry.valid(entt::entity{2}));
  427. ASSERT_EQ(e3, entt::entity{3});
  428. registry.release(e2);
  429. ASSERT_EQ(traits_type::to_version(e2), 0u);
  430. ASSERT_EQ(registry.current(e2), 1u);
  431. e2 = registry.create();
  432. auto e1 = registry.create(entt::entity{2});
  433. ASSERT_EQ(traits_type::to_entity(e2), 1u);
  434. ASSERT_EQ(traits_type::to_version(e2), 1u);
  435. ASSERT_EQ(traits_type::to_entity(e1), 2u);
  436. ASSERT_EQ(traits_type::to_version(e1), 0u);
  437. registry.release(e1);
  438. registry.release(e2);
  439. auto e0 = registry.create(entt::entity{0});
  440. ASSERT_EQ(e0, entt::entity{0});
  441. ASSERT_EQ(traits_type::to_version(e0), 0u);
  442. }
  443. TEST(Registry, CreateClearCycle) {
  444. using traits_type = entt::entt_traits<entt::entity>;
  445. entt::registry registry;
  446. entt::entity pre{}, post{};
  447. for(int i = 0; i < 10; ++i) {
  448. const auto entity = registry.create();
  449. registry.emplace<double>(entity);
  450. }
  451. registry.clear();
  452. for(int i = 0; i < 7; ++i) {
  453. const auto entity = registry.create();
  454. registry.emplace<int>(entity);
  455. if(i == 3) {
  456. pre = entity;
  457. }
  458. }
  459. registry.clear();
  460. for(int i = 0; i < 5; ++i) {
  461. const auto entity = registry.create();
  462. if(i == 3) {
  463. post = entity;
  464. }
  465. }
  466. ASSERT_FALSE(registry.valid(pre));
  467. ASSERT_TRUE(registry.valid(post));
  468. ASSERT_NE(traits_type::to_version(pre), traits_type::to_version(post));
  469. ASSERT_EQ(traits_type::to_version(pre) + 1, traits_type::to_version(post));
  470. ASSERT_EQ(registry.current(pre), registry.current(post));
  471. }
  472. TEST(Registry, CreateDestroyReleaseCornerCase) {
  473. entt::registry registry;
  474. const auto e0 = registry.create();
  475. const auto e1 = registry.create();
  476. registry.destroy(e0);
  477. registry.release(e1);
  478. registry.each([](auto) { FAIL(); });
  479. ASSERT_EQ(registry.current(e0), 1u);
  480. ASSERT_EQ(registry.current(e1), 1u);
  481. }
  482. ENTT_DEBUG_TEST(RegistryDeathTest, CreateTooManyEntities) {
  483. entt::basic_registry<small_entity> registry;
  484. std::vector<small_entity> entities(entt::entt_traits<small_entity>::to_entity(entt::null));
  485. registry.create(entities.begin(), entities.end());
  486. ASSERT_DEATH([[maybe_unused]] const auto entity = registry.create(), "");
  487. }
  488. TEST(Registry, DestroyVersion) {
  489. entt::registry registry;
  490. const auto e0 = registry.create();
  491. const auto e1 = registry.create();
  492. ASSERT_EQ(registry.current(e0), 0u);
  493. ASSERT_EQ(registry.current(e1), 0u);
  494. registry.destroy(e0);
  495. registry.destroy(e1, 3);
  496. ASSERT_EQ(registry.current(e0), 1u);
  497. ASSERT_EQ(registry.current(e1), 3u);
  498. }
  499. ENTT_DEBUG_TEST(RegistryDeathTest, DestroyVersion) {
  500. entt::registry registry;
  501. const auto entity = registry.create();
  502. registry.destroy(entity);
  503. ASSERT_DEATH(registry.destroy(entity), "");
  504. ASSERT_DEATH(registry.destroy(entity, 3), "");
  505. }
  506. TEST(Registry, DestroyRange) {
  507. entt::registry registry;
  508. const auto iview = registry.view<int>();
  509. const auto icview = registry.view<int, char>();
  510. entt::entity entities[3u];
  511. registry.create(std::begin(entities), std::end(entities));
  512. registry.emplace<int>(entities[0u]);
  513. registry.emplace<char>(entities[0u]);
  514. registry.emplace<double>(entities[0u]);
  515. registry.emplace<int>(entities[1u]);
  516. registry.emplace<char>(entities[1u]);
  517. registry.emplace<int>(entities[2u]);
  518. ASSERT_TRUE(registry.valid(entities[0u]));
  519. ASSERT_TRUE(registry.valid(entities[1u]));
  520. ASSERT_TRUE(registry.valid(entities[2u]));
  521. registry.destroy(icview.begin(), icview.end());
  522. ASSERT_FALSE(registry.valid(entities[0u]));
  523. ASSERT_FALSE(registry.valid(entities[1u]));
  524. ASSERT_TRUE(registry.valid(entities[2u]));
  525. ASSERT_EQ(registry.storage<int>().size(), 1u);
  526. ASSERT_EQ(registry.storage<char>().size(), 0u);
  527. ASSERT_EQ(registry.storage<double>().size(), 0u);
  528. registry.destroy(iview.begin(), iview.end());
  529. ASSERT_FALSE(registry.valid(entities[2u]));
  530. ASSERT_NO_FATAL_FAILURE(registry.destroy(iview.rbegin(), iview.rend()));
  531. ASSERT_EQ(iview.size(), 0u);
  532. ASSERT_EQ(icview.size_hint(), 0u);
  533. ASSERT_EQ(registry.storage<int>().size(), 0u);
  534. ASSERT_EQ(registry.storage<char>().size(), 0u);
  535. ASSERT_EQ(registry.storage<double>().size(), 0u);
  536. registry.create(std::begin(entities), std::end(entities));
  537. registry.insert<int>(std::begin(entities), std::end(entities));
  538. ASSERT_TRUE(registry.valid(entities[0u]));
  539. ASSERT_TRUE(registry.valid(entities[1u]));
  540. ASSERT_TRUE(registry.valid(entities[2u]));
  541. ASSERT_EQ(registry.storage<int>().size(), 3u);
  542. registry.destroy(std::begin(entities), std::end(entities));
  543. ASSERT_FALSE(registry.valid(entities[0u]));
  544. ASSERT_FALSE(registry.valid(entities[1u]));
  545. ASSERT_FALSE(registry.valid(entities[2u]));
  546. ASSERT_EQ(registry.storage<int>().size(), 0u);
  547. entt::sparse_set managed{};
  548. registry.create(std::begin(entities), std::end(entities));
  549. managed.push(std::begin(entities), std::end(entities));
  550. registry.insert<int>(managed.begin(), managed.end());
  551. ASSERT_TRUE(registry.valid(managed[0u]));
  552. ASSERT_TRUE(registry.valid(managed[1u]));
  553. ASSERT_TRUE(registry.valid(managed[2u]));
  554. ASSERT_EQ(registry.storage<int>().size(), 3u);
  555. registry.destroy(managed.begin(), managed.end());
  556. ASSERT_FALSE(registry.valid(managed[0u]));
  557. ASSERT_FALSE(registry.valid(managed[1u]));
  558. ASSERT_FALSE(registry.valid(managed[2u]));
  559. ASSERT_EQ(registry.storage<int>().size(), 0u);
  560. }
  561. TEST(Registry, StableDestroy) {
  562. entt::registry registry;
  563. const auto iview = registry.view<int>();
  564. const auto icview = registry.view<int, stable_type>();
  565. entt::entity entities[3u];
  566. registry.create(std::begin(entities), std::end(entities));
  567. registry.emplace<int>(entities[0u]);
  568. registry.emplace<stable_type>(entities[0u]);
  569. registry.emplace<double>(entities[0u]);
  570. registry.emplace<int>(entities[1u]);
  571. registry.emplace<stable_type>(entities[1u]);
  572. registry.emplace<int>(entities[2u]);
  573. ASSERT_TRUE(registry.valid(entities[0u]));
  574. ASSERT_TRUE(registry.valid(entities[1u]));
  575. ASSERT_TRUE(registry.valid(entities[2u]));
  576. registry.destroy(icview.begin(), icview.end());
  577. ASSERT_FALSE(registry.valid(entities[0u]));
  578. ASSERT_FALSE(registry.valid(entities[1u]));
  579. ASSERT_TRUE(registry.valid(entities[2u]));
  580. ASSERT_EQ(registry.storage<int>().size(), 1u);
  581. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  582. ASSERT_EQ(registry.storage<double>().size(), 0u);
  583. registry.destroy(iview.begin(), iview.end());
  584. ASSERT_FALSE(registry.valid(entities[2u]));
  585. ASSERT_EQ(iview.size(), 0u);
  586. ASSERT_EQ(icview.size_hint(), 0u);
  587. ASSERT_EQ(registry.storage<int>().size(), 0u);
  588. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  589. ASSERT_EQ(registry.storage<double>().size(), 0u);
  590. }
  591. TEST(Registry, ReleaseVersion) {
  592. entt::registry registry;
  593. entt::entity entities[2u];
  594. registry.create(std::begin(entities), std::end(entities));
  595. ASSERT_EQ(registry.current(entities[0u]), 0u);
  596. ASSERT_EQ(registry.current(entities[1u]), 0u);
  597. registry.release(entities[0u]);
  598. registry.release(entities[1u], 3);
  599. ASSERT_EQ(registry.current(entities[0u]), 1u);
  600. ASSERT_EQ(registry.current(entities[1u]), 3u);
  601. }
  602. ENTT_DEBUG_TEST(RegistryDeathTest, ReleaseVersion) {
  603. entt::registry registry;
  604. entt::entity entity = registry.create();
  605. registry.release(entity);
  606. ASSERT_DEATH(registry.release(entity), "");
  607. ASSERT_DEATH(registry.release(entity, 3), "");
  608. }
  609. TEST(Registry, ReleaseRange) {
  610. entt::registry registry;
  611. entt::entity entities[3u];
  612. registry.create(std::begin(entities), std::end(entities));
  613. ASSERT_TRUE(registry.valid(entities[0u]));
  614. ASSERT_TRUE(registry.valid(entities[1u]));
  615. ASSERT_TRUE(registry.valid(entities[2u]));
  616. registry.release(std::begin(entities), std::end(entities) - 1u);
  617. ASSERT_FALSE(registry.valid(entities[0u]));
  618. ASSERT_FALSE(registry.valid(entities[1u]));
  619. ASSERT_TRUE(registry.valid(entities[2u]));
  620. registry.release(std::end(entities) - 1u, std::end(entities));
  621. ASSERT_FALSE(registry.valid(entities[2u]));
  622. }
  623. TEST(Registry, VersionOverflow) {
  624. using traits_type = entt::entt_traits<entt::entity>;
  625. entt::registry registry;
  626. const auto entity = registry.create();
  627. registry.release(entity);
  628. ASSERT_NE(registry.current(entity), traits_type::to_version(entity));
  629. ASSERT_NE(registry.current(entity), typename traits_type::version_type{});
  630. registry.release(registry.create(), traits_type::to_version(entt::tombstone) - 1u);
  631. registry.release(registry.create());
  632. ASSERT_EQ(registry.current(entity), traits_type::to_version(entity));
  633. ASSERT_EQ(registry.current(entity), typename traits_type::version_type{});
  634. }
  635. TEST(Registry, NullEntity) {
  636. entt::registry registry;
  637. const entt::entity entity = entt::null;
  638. ASSERT_FALSE(registry.valid(entity));
  639. ASSERT_NE(registry.create(entity), entity);
  640. }
  641. TEST(Registry, TombstoneVersion) {
  642. using traits_type = entt::entt_traits<entt::entity>;
  643. entt::registry registry;
  644. const entt::entity entity = entt::tombstone;
  645. ASSERT_FALSE(registry.valid(entity));
  646. const auto other = registry.create();
  647. const auto vers = traits_type::to_version(entity);
  648. const auto required = traits_type::construct(traits_type::to_entity(other), vers);
  649. ASSERT_NE(registry.release(other, vers), vers);
  650. ASSERT_NE(registry.create(required), required);
  651. }
  652. TEST(Registry, Each) {
  653. entt::registry registry;
  654. entt::registry::size_type tot;
  655. entt::registry::size_type match;
  656. static_cast<void>(registry.create());
  657. registry.emplace<int>(registry.create());
  658. static_cast<void>(registry.create());
  659. registry.emplace<int>(registry.create());
  660. static_cast<void>(registry.create());
  661. tot = 0u;
  662. match = 0u;
  663. registry.each([&](auto entity) {
  664. match += registry.all_of<int>(entity);
  665. static_cast<void>(registry.create());
  666. ++tot;
  667. });
  668. ASSERT_EQ(tot, 5u);
  669. ASSERT_EQ(match, 2u);
  670. tot = 0u;
  671. match = 0u;
  672. registry.each([&](auto entity) {
  673. if(registry.all_of<int>(entity)) {
  674. registry.destroy(entity);
  675. ++match;
  676. }
  677. ++tot;
  678. });
  679. ASSERT_EQ(tot, 10u);
  680. ASSERT_EQ(match, 2u);
  681. tot = 0u;
  682. match = 0u;
  683. registry.each([&](auto entity) {
  684. match += registry.all_of<int>(entity);
  685. registry.destroy(entity);
  686. ++tot;
  687. });
  688. ASSERT_EQ(tot, 8u);
  689. ASSERT_EQ(match, 0u);
  690. registry.each([&](auto) { FAIL(); });
  691. }
  692. TEST(Registry, Orphans) {
  693. entt::registry registry;
  694. entt::entity entities[3u]{};
  695. registry.create(std::begin(entities), std::end(entities));
  696. registry.emplace<int>(entities[0u]);
  697. registry.emplace<int>(entities[2u]);
  698. registry.each([&](const auto entt) {
  699. ASSERT_TRUE(entt != entities[1u] || registry.orphan(entt));
  700. });
  701. registry.erase<int>(entities[0u]);
  702. registry.erase<int>(entities[2u]);
  703. registry.each([&](const auto entt) {
  704. ASSERT_TRUE(registry.orphan(entt));
  705. });
  706. }
  707. TEST(Registry, View) {
  708. entt::registry registry;
  709. entt::entity entities[3u];
  710. auto iview = registry.view<int>();
  711. auto cview = registry.view<char>();
  712. auto mview = registry.view<int, char>();
  713. auto fview = registry.view<int>(entt::exclude<char>);
  714. registry.create(std::begin(entities), std::end(entities));
  715. registry.emplace<int>(entities[0u], 0);
  716. registry.emplace<char>(entities[0u], 'c');
  717. registry.emplace<int>(entities[1u], 0);
  718. registry.emplace<int>(entities[2u], 0);
  719. registry.emplace<char>(entities[2u], 'c');
  720. ASSERT_EQ(iview.size(), 3u);
  721. ASSERT_EQ(cview.size(), 2u);
  722. ASSERT_EQ(mview.size_hint(), 3u);
  723. ASSERT_EQ(fview.size_hint(), 3u);
  724. mview.refresh();
  725. fview.refresh();
  726. ASSERT_EQ(mview.size_hint(), 2u);
  727. ASSERT_EQ(fview.size_hint(), 3u);
  728. ASSERT_NE(mview.begin(), mview.end());
  729. ASSERT_NE(fview.begin(), fview.end());
  730. ASSERT_EQ(std::distance(mview.begin(), mview.end()), 2);
  731. ASSERT_EQ(std::distance(fview.begin(), fview.end()), 1);
  732. mview.each([&entities, first = true](auto entity, auto &&...) mutable {
  733. ASSERT_EQ(entity, entities[2u * first]);
  734. first = false;
  735. });
  736. fview.each([&entities](auto entity, auto &&...) {
  737. ASSERT_EQ(entity, entities[1u]);
  738. });
  739. }
  740. TEST(Registry, ExcludeOnlyView) {
  741. entt::registry registry;
  742. entt::entity entities[4u];
  743. auto view = registry.view<entt::entity>(entt::exclude<int>);
  744. registry.create(std::begin(entities), std::end(entities));
  745. registry.emplace<int>(entities[0u], 0);
  746. registry.emplace<int>(entities[2u], 0);
  747. registry.emplace<int>(entities[3u], 0);
  748. registry.destroy(entities[3u]);
  749. ASSERT_EQ(view.size_hint(), 4u);
  750. ASSERT_NE(view.begin(), view.end());
  751. // returns all matching identifiers, both in-use and available ones
  752. ASSERT_EQ(std::distance(view.begin(), view.end()), 2);
  753. // skips available identifiers automatically, only returns in-use elements
  754. view.each([&entities](auto entity, auto &&...) {
  755. ASSERT_EQ(entity, entities[1u]);
  756. });
  757. }
  758. TEST(Registry, NonOwningGroupInitOnFirstUse) {
  759. entt::registry registry;
  760. entt::entity entities[3u];
  761. registry.create(std::begin(entities), std::end(entities));
  762. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  763. registry.emplace<char>(entities[0u], 'c');
  764. registry.emplace<char>(entities[2u], 'c');
  765. std::size_t cnt{};
  766. auto group = registry.group(entt::get<int, char>);
  767. group.each([&cnt](auto...) { ++cnt; });
  768. ASSERT_FALSE((registry.owned<int, char>()));
  769. ASSERT_EQ(cnt, 2u);
  770. }
  771. TEST(Registry, NonOwningGroupInitOnEmplace) {
  772. entt::registry registry;
  773. entt::entity entities[3u];
  774. auto group = registry.group(entt::get<int, char>);
  775. registry.create(std::begin(entities), std::end(entities));
  776. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  777. registry.emplace<char>(entities[0u], 'c');
  778. registry.emplace<char>(entities[2u], 'c');
  779. std::size_t cnt{};
  780. group.each([&cnt](auto...) { ++cnt; });
  781. ASSERT_FALSE((registry.owned<int, char>()));
  782. ASSERT_EQ(cnt, 2u);
  783. }
  784. TEST(Registry, FullOwningGroupInitOnFirstUse) {
  785. entt::registry registry;
  786. entt::entity entities[3u];
  787. registry.create(std::begin(entities), std::end(entities));
  788. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  789. registry.emplace<char>(entities[0u], 'c');
  790. registry.emplace<char>(entities[2u], 'c');
  791. std::size_t cnt{};
  792. auto group = registry.group<int, char>();
  793. group.each([&cnt](auto...) { ++cnt; });
  794. ASSERT_TRUE(registry.owned<int>());
  795. ASSERT_TRUE(registry.owned<char>());
  796. ASSERT_FALSE(registry.owned<double>());
  797. ASSERT_EQ(cnt, 2u);
  798. }
  799. TEST(Registry, FullOwningGroupInitOnEmplace) {
  800. entt::registry registry;
  801. entt::entity entities[3u];
  802. auto group = registry.group<int, char>();
  803. registry.create(std::begin(entities), std::end(entities));
  804. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  805. registry.emplace<char>(entities[0u], 'c');
  806. registry.emplace<char>(entities[2u], 'c');
  807. std::size_t cnt{};
  808. group.each([&cnt](auto...) { ++cnt; });
  809. ASSERT_TRUE(registry.owned<int>());
  810. ASSERT_TRUE(registry.owned<char>());
  811. ASSERT_FALSE(registry.owned<double>());
  812. ASSERT_EQ(cnt, 2u);
  813. }
  814. TEST(Registry, PartialOwningGroupInitOnFirstUse) {
  815. entt::registry registry;
  816. entt::entity entities[3u];
  817. registry.create(std::begin(entities), std::end(entities));
  818. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  819. registry.emplace<char>(entities[0u], 'c');
  820. registry.emplace<char>(entities[2u], 'c');
  821. std::size_t cnt{};
  822. auto group = registry.group<int>(entt::get<char>);
  823. group.each([&cnt](auto...) { ++cnt; });
  824. ASSERT_TRUE((registry.owned<int, char>()));
  825. ASSERT_TRUE(registry.owned<int>());
  826. ASSERT_FALSE(registry.owned<char>());
  827. ASSERT_EQ(cnt, 2u);
  828. }
  829. TEST(Registry, PartialOwningGroupInitOnEmplace) {
  830. entt::registry registry;
  831. entt::entity entities[3u];
  832. auto group = registry.group<int>(entt::get<char>);
  833. registry.create(std::begin(entities), std::end(entities));
  834. registry.insert<int>(std::begin(entities), std::end(entities), 0);
  835. registry.emplace<char>(entities[0u], 'c');
  836. registry.emplace<char>(entities[2u], 'c');
  837. std::size_t cnt{};
  838. group.each([&cnt](auto...) { ++cnt; });
  839. ASSERT_TRUE((registry.owned<int, char>()));
  840. ASSERT_TRUE(registry.owned<int>());
  841. ASSERT_FALSE(registry.owned<char>());
  842. ASSERT_EQ(cnt, 2u);
  843. }
  844. TEST(Registry, CleanViewAfterRemoveAndClear) {
  845. entt::registry registry;
  846. auto view = registry.view<int, char>();
  847. const auto entity = registry.create();
  848. registry.emplace<int>(entity);
  849. registry.emplace<char>(entity);
  850. ASSERT_EQ(view.size_hint(), 1u);
  851. registry.erase<char>(entity);
  852. ASSERT_EQ(view.size_hint(), 1u);
  853. registry.emplace<char>(entity);
  854. ASSERT_EQ(view.size_hint(), 1u);
  855. registry.clear<int>();
  856. ASSERT_EQ(view.size_hint(), 0u);
  857. registry.emplace<int>(entity);
  858. ASSERT_EQ(view.size_hint(), 1u);
  859. registry.clear();
  860. ASSERT_EQ(view.size_hint(), 0u);
  861. }
  862. TEST(Registry, CleanNonOwningGroupViewAfterRemoveAndClear) {
  863. entt::registry registry;
  864. auto group = registry.group(entt::get<int, char>);
  865. const auto entity = registry.create();
  866. registry.emplace<int>(entity, 0);
  867. registry.emplace<char>(entity, 'c');
  868. ASSERT_EQ(group.size(), 1u);
  869. registry.erase<char>(entity);
  870. ASSERT_EQ(group.size(), 0u);
  871. registry.emplace<char>(entity, 'c');
  872. ASSERT_EQ(group.size(), 1u);
  873. registry.clear<int>();
  874. ASSERT_EQ(group.size(), 0u);
  875. registry.emplace<int>(entity, 0);
  876. ASSERT_EQ(group.size(), 1u);
  877. registry.clear();
  878. ASSERT_EQ(group.size(), 0u);
  879. }
  880. TEST(Registry, CleanFullOwningGroupViewAfterRemoveAndClear) {
  881. entt::registry registry;
  882. auto group = registry.group<int, char>();
  883. const auto entity = registry.create();
  884. registry.emplace<int>(entity, 0);
  885. registry.emplace<char>(entity, 'c');
  886. ASSERT_EQ(group.size(), 1u);
  887. registry.erase<char>(entity);
  888. ASSERT_EQ(group.size(), 0u);
  889. registry.emplace<char>(entity, 'c');
  890. ASSERT_EQ(group.size(), 1u);
  891. registry.clear<int>();
  892. ASSERT_EQ(group.size(), 0u);
  893. registry.emplace<int>(entity, 0);
  894. ASSERT_EQ(group.size(), 1u);
  895. registry.clear();
  896. ASSERT_EQ(group.size(), 0u);
  897. }
  898. TEST(Registry, CleanPartialOwningGroupViewAfterRemoveAndClear) {
  899. entt::registry registry;
  900. auto group = registry.group<int>(entt::get<char>);
  901. const auto entity = registry.create();
  902. registry.emplace<int>(entity, 0);
  903. registry.emplace<char>(entity, 'c');
  904. ASSERT_EQ(group.size(), 1u);
  905. registry.erase<char>(entity);
  906. ASSERT_EQ(group.size(), 0u);
  907. registry.emplace<char>(entity, 'c');
  908. ASSERT_EQ(group.size(), 1u);
  909. registry.clear<int>();
  910. ASSERT_EQ(group.size(), 0u);
  911. registry.emplace<int>(entity, 0);
  912. ASSERT_EQ(group.size(), 1u);
  913. registry.clear();
  914. ASSERT_EQ(group.size(), 0u);
  915. }
  916. ENTT_DEBUG_TEST(RegistryDeathTest, NestedGroups) {
  917. entt::registry registry;
  918. registry.group<int, double>(entt::get<char>);
  919. ASSERT_DEATH(registry.group<int>(entt::get<char>), "");
  920. ASSERT_DEATH(registry.group<int>(entt::get<char, double>), "");
  921. ASSERT_DEATH(registry.group<int>(entt::get<char>, entt::exclude<double>), "");
  922. ASSERT_DEATH((registry.group<int, double>()), "");
  923. }
  924. ENTT_DEBUG_TEST(RegistryDeathTest, ConflictingGroups) {
  925. entt::registry registry;
  926. registry.group<char>(entt::get<int>, entt::exclude<double>);
  927. ASSERT_DEATH(registry.group<char>(entt::get<float>, entt::exclude<double>), "");
  928. }
  929. TEST(Registry, SortSingle) {
  930. entt::registry registry;
  931. int val = 0;
  932. registry.emplace<int>(registry.create(), val++);
  933. registry.emplace<int>(registry.create(), val++);
  934. registry.emplace<int>(registry.create(), val++);
  935. for(auto entity: registry.view<int>()) {
  936. ASSERT_EQ(registry.get<int>(entity), --val);
  937. }
  938. registry.sort<int>(std::less<int>{});
  939. for(auto entity: registry.view<int>()) {
  940. ASSERT_EQ(registry.get<int>(entity), val++);
  941. }
  942. }
  943. TEST(Registry, SortMulti) {
  944. entt::registry registry;
  945. unsigned int uval = 0u;
  946. int ival = 0;
  947. for(auto i = 0; i < 3; ++i) {
  948. const auto entity = registry.create();
  949. registry.emplace<unsigned int>(entity, uval++);
  950. registry.emplace<int>(entity, ival++);
  951. }
  952. for(auto entity: registry.view<unsigned int>()) {
  953. ASSERT_EQ(registry.get<unsigned int>(entity), --uval);
  954. }
  955. for(auto entity: registry.view<int>()) {
  956. ASSERT_EQ(registry.get<int>(entity), --ival);
  957. }
  958. registry.sort<unsigned int>(std::less<unsigned int>{});
  959. registry.sort<int, unsigned int>();
  960. for(auto entity: registry.view<unsigned int>()) {
  961. ASSERT_EQ(registry.get<unsigned int>(entity), uval++);
  962. }
  963. for(auto entity: registry.view<int>()) {
  964. ASSERT_EQ(registry.get<int>(entity), ival++);
  965. }
  966. }
  967. TEST(Registry, SortEmpty) {
  968. entt::registry registry;
  969. registry.emplace<empty_type>(registry.create());
  970. registry.emplace<empty_type>(registry.create());
  971. registry.emplace<empty_type>(registry.create());
  972. ASSERT_LT(registry.storage<empty_type>().data()[0], registry.storage<empty_type>().data()[1]);
  973. ASSERT_LT(registry.storage<empty_type>().data()[1], registry.storage<empty_type>().data()[2]);
  974. registry.sort<empty_type>(std::less<entt::entity>{});
  975. ASSERT_GT(registry.storage<empty_type>().data()[0], registry.storage<empty_type>().data()[1]);
  976. ASSERT_GT(registry.storage<empty_type>().data()[1], registry.storage<empty_type>().data()[2]);
  977. }
  978. TEST(Registry, ComponentsWithTypesFromStandardTemplateLibrary) {
  979. // see #37 - the test shouldn't crash, that's all
  980. entt::registry registry;
  981. const auto entity = registry.create();
  982. registry.emplace<std::unordered_set<int>>(entity).insert(42);
  983. registry.destroy(entity);
  984. }
  985. TEST(Registry, ConstructWithComponents) {
  986. // it should compile, that's all
  987. entt::registry registry;
  988. const auto value = 0;
  989. registry.emplace<int>(registry.create(), value);
  990. }
  991. TEST(Registry, Signals) {
  992. entt::registry registry;
  993. entt::entity entities[2u];
  994. listener listener;
  995. registry.on_construct<empty_type>().connect<&listener::incr>(listener);
  996. registry.on_destroy<empty_type>().connect<&listener::decr>(listener);
  997. registry.on_construct<int>().connect<&listener::incr>(listener);
  998. registry.on_destroy<int>().connect<&listener::decr>(listener);
  999. registry.create(std::begin(entities), std::end(entities));
  1000. registry.insert<empty_type>(std::begin(entities), std::end(entities));
  1001. ASSERT_EQ(listener.counter, 2);
  1002. ASSERT_EQ(listener.last, entities[1u]);
  1003. registry.insert<int>(std::rbegin(entities), std::rend(entities));
  1004. ASSERT_EQ(listener.counter, 4);
  1005. ASSERT_EQ(listener.last, entities[0u]);
  1006. registry.erase<empty_type, int>(entities[0u]);
  1007. ASSERT_EQ(listener.counter, 2);
  1008. ASSERT_EQ(listener.last, entities[0u]);
  1009. registry.on_destroy<empty_type>().disconnect<&listener::decr>(listener);
  1010. registry.on_destroy<int>().disconnect<&listener::decr>(listener);
  1011. registry.erase<empty_type, int>(entities[1u]);
  1012. ASSERT_EQ(listener.counter, 2);
  1013. ASSERT_EQ(listener.last, entities[0u]);
  1014. registry.on_construct<empty_type>().disconnect<&listener::incr>(listener);
  1015. registry.on_construct<int>().disconnect<&listener::incr>(listener);
  1016. registry.emplace<empty_type>(entities[1u]);
  1017. registry.emplace<int>(entities[1u]);
  1018. ASSERT_EQ(listener.counter, 2);
  1019. ASSERT_EQ(listener.last, entities[0u]);
  1020. registry.on_construct<int>().connect<&listener::incr>(listener);
  1021. registry.on_destroy<int>().connect<&listener::decr>(listener);
  1022. registry.emplace<int>(entities[0u]);
  1023. registry.erase<int>(entities[1u]);
  1024. ASSERT_EQ(listener.counter, 2);
  1025. ASSERT_EQ(listener.last, entities[1u]);
  1026. registry.on_construct<empty_type>().connect<&listener::incr>(listener);
  1027. registry.on_destroy<empty_type>().connect<&listener::decr>(listener);
  1028. registry.erase<empty_type>(entities[1u]);
  1029. registry.emplace<empty_type>(entities[0u]);
  1030. ASSERT_EQ(listener.counter, 2);
  1031. ASSERT_EQ(listener.last, entities[0u]);
  1032. registry.clear<empty_type, int>();
  1033. ASSERT_EQ(listener.counter, 0);
  1034. ASSERT_EQ(listener.last, entities[0u]);
  1035. registry.insert<empty_type>(std::begin(entities), std::end(entities));
  1036. registry.insert<int>(std::begin(entities), std::end(entities));
  1037. registry.destroy(entities[1u]);
  1038. ASSERT_EQ(listener.counter, 2);
  1039. ASSERT_EQ(listener.last, entities[1u]);
  1040. registry.erase<int, empty_type>(entities[0u]);
  1041. registry.emplace_or_replace<int>(entities[0u]);
  1042. registry.emplace_or_replace<empty_type>(entities[0u]);
  1043. ASSERT_EQ(listener.counter, 2);
  1044. ASSERT_EQ(listener.last, entities[0u]);
  1045. registry.on_destroy<empty_type>().disconnect<&listener::decr>(listener);
  1046. registry.on_destroy<int>().disconnect<&listener::decr>(listener);
  1047. registry.emplace_or_replace<empty_type>(entities[0u]);
  1048. registry.emplace_or_replace<int>(entities[0u]);
  1049. ASSERT_EQ(listener.counter, 2);
  1050. ASSERT_EQ(listener.last, entities[0u]);
  1051. registry.on_update<empty_type>().connect<&listener::incr>(listener);
  1052. registry.on_update<int>().connect<&listener::incr>(listener);
  1053. registry.emplace_or_replace<empty_type>(entities[0u]);
  1054. registry.emplace_or_replace<int>(entities[0u]);
  1055. ASSERT_EQ(listener.counter, 4);
  1056. ASSERT_EQ(listener.last, entities[0u]);
  1057. registry.replace<empty_type>(entities[0u]);
  1058. registry.replace<int>(entities[0u]);
  1059. ASSERT_EQ(listener.counter, 6);
  1060. ASSERT_EQ(listener.last, entities[0u]);
  1061. }
  1062. TEST(Registry, SignalsOnRuntimePool) {
  1063. using namespace entt::literals;
  1064. entt::registry registry;
  1065. const auto entity = registry.create();
  1066. listener listener;
  1067. registry.on_construct<int>("custom"_hs).connect<&listener::incr>(listener);
  1068. registry.on_update<int>("custom"_hs).connect<&listener::incr>(listener);
  1069. registry.on_destroy<int>("custom"_hs).connect<&listener::incr>(listener);
  1070. ASSERT_EQ(listener.counter, 0);
  1071. registry.emplace<int>(entity);
  1072. registry.patch<int>(entity);
  1073. registry.erase<int>(entity);
  1074. ASSERT_EQ(listener.counter, 0);
  1075. registry.storage<int>("custom"_hs).emplace(entity);
  1076. registry.storage<int>("custom"_hs).patch(entity);
  1077. registry.storage<int>("custom"_hs).erase(entity);
  1078. ASSERT_EQ(listener.counter, 3);
  1079. }
  1080. TEST(Registry, SignalsOnEntity) {
  1081. entt::registry registry;
  1082. listener listener;
  1083. registry.on_construct<entt::entity>().connect<&listener::incr>(listener);
  1084. entt::entity entity = registry.create();
  1085. entt::entity other = registry.create();
  1086. ASSERT_EQ(listener.counter, 2);
  1087. ASSERT_EQ(listener.last, other);
  1088. registry.destroy(other);
  1089. registry.destroy(entity);
  1090. ASSERT_EQ(listener.counter, 2);
  1091. ASSERT_EQ(listener.last, other);
  1092. registry.on_construct<entt::entity>().disconnect(&listener);
  1093. other = registry.create();
  1094. entity = registry.create();
  1095. ASSERT_EQ(listener.counter, 2);
  1096. ASSERT_NE(listener.last, entity);
  1097. ASSERT_NE(listener.last, other);
  1098. registry.on_update<entt::entity>().connect<&listener::decr>(listener);
  1099. registry.patch<entt::entity>(entity);
  1100. ASSERT_EQ(listener.counter, 1);
  1101. ASSERT_EQ(listener.last, entity);
  1102. registry.on_update<entt::entity>().disconnect(&listener);
  1103. registry.patch<entt::entity>(other);
  1104. ASSERT_EQ(listener.counter, 1);
  1105. ASSERT_NE(listener.last, other);
  1106. registry.on_destroy<entt::entity>().connect<&listener::decr>(listener);
  1107. registry.destroy(entity);
  1108. ASSERT_EQ(listener.counter, 0);
  1109. ASSERT_EQ(listener.last, entity);
  1110. registry.on_destroy<entt::entity>().disconnect(&listener);
  1111. registry.destroy(other);
  1112. ASSERT_EQ(listener.counter, 0);
  1113. ASSERT_NE(listener.last, other);
  1114. }
  1115. TEST(Registry, SignalWhenDestroying) {
  1116. entt::registry registry;
  1117. const auto entity = registry.create();
  1118. registry.on_destroy<double>().connect<&entt::registry::remove<char>>();
  1119. registry.emplace<double>(entity);
  1120. registry.emplace<int>(entity);
  1121. ASSERT_NE(registry.storage(entt::type_id<double>().hash()), nullptr);
  1122. ASSERT_NE(registry.storage(entt::type_id<int>().hash()), nullptr);
  1123. ASSERT_EQ(registry.storage(entt::type_id<char>().hash()), nullptr);
  1124. ASSERT_TRUE(registry.valid(entity));
  1125. registry.destroy(entity);
  1126. ASSERT_NE(registry.storage(entt::type_id<char>().hash()), nullptr);
  1127. ASSERT_FALSE(registry.valid(entity));
  1128. }
  1129. TEST(Registry, Insert) {
  1130. entt::registry registry;
  1131. entt::entity entities[3u];
  1132. registry.create(std::begin(entities), std::end(entities));
  1133. registry.emplace<int>(entities[0u]);
  1134. registry.emplace<char>(entities[0u]);
  1135. registry.emplace<double>(entities[0u]);
  1136. registry.emplace<int>(entities[1u]);
  1137. registry.emplace<char>(entities[1u]);
  1138. registry.emplace<int>(entities[2u]);
  1139. ASSERT_FALSE(registry.all_of<float>(entities[0u]));
  1140. ASSERT_FALSE(registry.all_of<float>(entities[1u]));
  1141. ASSERT_FALSE(registry.all_of<float>(entities[2u]));
  1142. const auto icview = registry.view<int, char>();
  1143. registry.insert(icview.begin(), icview.end(), 3.f);
  1144. ASSERT_EQ(registry.get<float>(entities[0u]), 3.f);
  1145. ASSERT_EQ(registry.get<float>(entities[1u]), 3.f);
  1146. ASSERT_FALSE(registry.all_of<float>(entities[2u]));
  1147. registry.clear<float>();
  1148. float value[3]{0.f, 1.f, 2.f};
  1149. const auto iview = registry.view<int>();
  1150. registry.insert<float>(iview.rbegin(), iview.rend(), value);
  1151. ASSERT_EQ(registry.get<float>(entities[0u]), 0.f);
  1152. ASSERT_EQ(registry.get<float>(entities[1u]), 1.f);
  1153. ASSERT_EQ(registry.get<float>(entities[2u]), 2.f);
  1154. }
  1155. TEST(Registry, Erase) {
  1156. entt::registry registry;
  1157. const auto iview = registry.view<int>();
  1158. const auto icview = registry.view<int, char>();
  1159. entt::entity entities[3u];
  1160. registry.create(std::begin(entities), std::end(entities));
  1161. registry.emplace<int>(entities[0u]);
  1162. registry.emplace<char>(entities[0u]);
  1163. registry.emplace<double>(entities[0u]);
  1164. registry.emplace<int>(entities[1u]);
  1165. registry.emplace<char>(entities[1u]);
  1166. registry.emplace<int>(entities[2u]);
  1167. ASSERT_TRUE(registry.any_of<int>(entities[0u]));
  1168. ASSERT_TRUE(registry.all_of<int>(entities[1u]));
  1169. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1170. registry.erase<int, char>(entities[0u]);
  1171. registry.erase<int, char>(icview.begin(), icview.end());
  1172. ASSERT_FALSE(registry.any_of<int>(entities[0u]));
  1173. ASSERT_FALSE(registry.all_of<int>(entities[1u]));
  1174. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1175. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1176. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1177. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1178. registry.erase<int>(iview.begin(), iview.end());
  1179. ASSERT_FALSE(registry.any_of<int>(entities[2u]));
  1180. ASSERT_NO_FATAL_FAILURE(registry.erase<int>(iview.rbegin(), iview.rend()));
  1181. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1182. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1183. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1184. registry.insert<int>(std::begin(entities) + 1, std::end(entities) - 1u);
  1185. registry.insert<char>(std::begin(entities) + 1, std::end(entities) - 1u);
  1186. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1187. ASSERT_EQ(registry.storage<char>().size(), 1u);
  1188. registry.erase<int, char>(iview.begin(), iview.end());
  1189. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1190. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1191. registry.insert<int>(std::begin(entities), std::end(entities));
  1192. registry.insert<char>(std::begin(entities), std::end(entities));
  1193. ASSERT_EQ(registry.storage<int>().size(), 3u);
  1194. ASSERT_EQ(registry.storage<char>().size(), 3u);
  1195. registry.erase<int, char>(std::begin(entities), std::end(entities));
  1196. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1197. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1198. ASSERT_FALSE(registry.orphan(entities[0u]));
  1199. ASSERT_TRUE(registry.orphan(entities[1u]));
  1200. ASSERT_TRUE(registry.orphan(entities[2u]));
  1201. }
  1202. ENTT_DEBUG_TEST(RegistryDeathTest, Erase) {
  1203. entt::registry registry;
  1204. const entt::entity entities[1u]{registry.create()};
  1205. ASSERT_FALSE((registry.any_of<int>(entities[0u])));
  1206. ASSERT_DEATH((registry.erase<int>(std::begin(entities), std::end(entities))), "");
  1207. ASSERT_DEATH(registry.erase<int>(entities[0u]), "");
  1208. }
  1209. TEST(Registry, StableErase) {
  1210. entt::registry registry;
  1211. const auto iview = registry.view<int>();
  1212. const auto icview = registry.view<int, stable_type>();
  1213. entt::entity entities[3u];
  1214. registry.create(std::begin(entities), std::end(entities));
  1215. registry.emplace<int>(entities[0u]);
  1216. registry.emplace<stable_type>(entities[0u]);
  1217. registry.emplace<double>(entities[0u]);
  1218. registry.emplace<int>(entities[1u]);
  1219. registry.emplace<stable_type>(entities[1u]);
  1220. registry.emplace<int>(entities[2u]);
  1221. ASSERT_TRUE(registry.any_of<int>(entities[0u]));
  1222. ASSERT_TRUE(registry.all_of<int>(entities[1u]));
  1223. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1224. registry.erase<int, stable_type>(entities[0u]);
  1225. registry.erase<int, stable_type>(icview.begin(), icview.end());
  1226. registry.erase<int, stable_type>(icview.begin(), icview.end());
  1227. ASSERT_FALSE(registry.any_of<int>(entities[0u]));
  1228. ASSERT_FALSE(registry.all_of<int>(entities[1u]));
  1229. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1230. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1231. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1232. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1233. registry.erase<int>(iview.begin(), iview.end());
  1234. ASSERT_FALSE(registry.any_of<int>(entities[2u]));
  1235. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1236. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1237. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1238. }
  1239. TEST(Registry, EraseIf) {
  1240. using namespace entt::literals;
  1241. entt::registry registry;
  1242. const auto entity = registry.create();
  1243. registry.emplace<int>(entity);
  1244. registry.storage<int>("other"_hs).emplace(entity);
  1245. registry.emplace<char>(entity);
  1246. ASSERT_TRUE(registry.storage<int>().contains(entity));
  1247. ASSERT_TRUE(registry.storage<int>("other"_hs).contains(entity));
  1248. ASSERT_TRUE(registry.storage<char>().contains(entity));
  1249. registry.erase_if(entity, [](auto &&...) { return false; });
  1250. ASSERT_TRUE(registry.storage<int>().contains(entity));
  1251. ASSERT_TRUE(registry.storage<int>("other"_hs).contains(entity));
  1252. ASSERT_TRUE(registry.storage<char>().contains(entity));
  1253. registry.erase_if(entity, [](entt::id_type id, auto &&...) { return id == "other"_hs; });
  1254. ASSERT_TRUE(registry.storage<int>().contains(entity));
  1255. ASSERT_FALSE(registry.storage<int>("other"_hs).contains(entity));
  1256. ASSERT_TRUE(registry.storage<char>().contains(entity));
  1257. registry.erase_if(entity, [](auto, const auto &storage) { return storage.type() == entt::type_id<char>(); });
  1258. ASSERT_TRUE(registry.storage<int>().contains(entity));
  1259. ASSERT_FALSE(registry.storage<int>("other"_hs).contains(entity));
  1260. ASSERT_FALSE(registry.storage<char>().contains(entity));
  1261. }
  1262. TEST(Registry, Remove) {
  1263. entt::registry registry;
  1264. const auto iview = registry.view<int>();
  1265. const auto icview = registry.view<int, char>();
  1266. entt::entity entities[3u];
  1267. registry.create(std::begin(entities), std::end(entities));
  1268. registry.emplace<int>(entities[0u]);
  1269. registry.emplace<char>(entities[0u]);
  1270. registry.emplace<double>(entities[0u]);
  1271. registry.emplace<int>(entities[1u]);
  1272. registry.emplace<char>(entities[1u]);
  1273. registry.emplace<int>(entities[2u]);
  1274. ASSERT_TRUE(registry.any_of<int>(entities[0u]));
  1275. ASSERT_TRUE(registry.all_of<int>(entities[1u]));
  1276. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1277. registry.remove<int, char>(entities[0u]);
  1278. ASSERT_EQ((registry.remove<int, char>(icview.begin(), icview.end())), 2u);
  1279. ASSERT_EQ((registry.remove<int, char>(icview.begin(), icview.end())), 0u);
  1280. ASSERT_FALSE(registry.any_of<int>(entities[0u]));
  1281. ASSERT_FALSE(registry.all_of<int>(entities[1u]));
  1282. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1283. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1284. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1285. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1286. ASSERT_EQ((registry.remove<int>(iview.begin(), iview.end())), 1u);
  1287. ASSERT_EQ(registry.remove<int>(entities[0u]), 0u);
  1288. ASSERT_EQ(registry.remove<int>(entities[1u]), 0u);
  1289. ASSERT_FALSE(registry.any_of<int>(entities[2u]));
  1290. ASSERT_EQ(registry.remove<int>(iview.begin(), iview.end()), 0u);
  1291. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1292. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1293. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1294. registry.insert<int>(std::begin(entities) + 1, std::end(entities) - 1u);
  1295. registry.insert<char>(std::begin(entities) + 1, std::end(entities) - 1u);
  1296. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1297. ASSERT_EQ(registry.storage<char>().size(), 1u);
  1298. registry.remove<int, char>(iview.begin(), iview.end());
  1299. registry.remove<int, char>(iview.begin(), iview.end());
  1300. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1301. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1302. registry.insert<int>(std::begin(entities), std::end(entities));
  1303. registry.insert<char>(std::begin(entities), std::end(entities));
  1304. ASSERT_EQ(registry.storage<int>().size(), 3u);
  1305. ASSERT_EQ(registry.storage<char>().size(), 3u);
  1306. registry.remove<int, char>(std::begin(entities), std::end(entities));
  1307. registry.remove<int, char>(std::begin(entities), std::end(entities));
  1308. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1309. ASSERT_EQ(registry.storage<char>().size(), 0u);
  1310. ASSERT_FALSE(registry.orphan(entities[0u]));
  1311. ASSERT_TRUE(registry.orphan(entities[1u]));
  1312. ASSERT_TRUE(registry.orphan(entities[2u]));
  1313. }
  1314. TEST(Registry, StableRemove) {
  1315. entt::registry registry;
  1316. const auto iview = registry.view<int>();
  1317. const auto icview = registry.view<int, stable_type>();
  1318. entt::entity entities[3u];
  1319. registry.create(std::begin(entities), std::end(entities));
  1320. registry.emplace<int>(entities[0u]);
  1321. registry.emplace<stable_type>(entities[0u]);
  1322. registry.emplace<double>(entities[0u]);
  1323. registry.emplace<int>(entities[1u]);
  1324. registry.emplace<stable_type>(entities[1u]);
  1325. registry.emplace<int>(entities[2u]);
  1326. ASSERT_TRUE(registry.any_of<int>(entities[0u]));
  1327. ASSERT_TRUE(registry.all_of<int>(entities[1u]));
  1328. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1329. registry.remove<int, stable_type>(entities[0u]);
  1330. ASSERT_EQ((registry.remove<int, stable_type>(icview.begin(), icview.end())), 2u);
  1331. ASSERT_EQ((registry.remove<int, stable_type>(icview.begin(), icview.end())), 0u);
  1332. ASSERT_FALSE(registry.any_of<int>(entities[0u]));
  1333. ASSERT_FALSE(registry.all_of<int>(entities[1u]));
  1334. ASSERT_TRUE(registry.any_of<int>(entities[2u]));
  1335. ASSERT_EQ(registry.storage<int>().size(), 1u);
  1336. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1337. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1338. ASSERT_EQ((registry.remove<int>(iview.begin(), iview.end())), 1u);
  1339. ASSERT_EQ(registry.remove<int>(entities[0u]), 0u);
  1340. ASSERT_EQ(registry.remove<int>(entities[1u]), 0u);
  1341. ASSERT_FALSE(registry.any_of<int>(entities[2u]));
  1342. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1343. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1344. ASSERT_EQ(registry.storage<double>().size(), 1u);
  1345. }
  1346. TEST(Registry, Compact) {
  1347. entt::registry registry;
  1348. entt::entity entities[2u];
  1349. registry.create(std::begin(entities), std::end(entities));
  1350. registry.emplace<int>(entities[0u]);
  1351. registry.emplace<stable_type>(entities[0u]);
  1352. registry.emplace<int>(entities[1u]);
  1353. registry.emplace<stable_type>(entities[1u]);
  1354. ASSERT_EQ(registry.storage<int>().size(), 2u);
  1355. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1356. registry.destroy(std::begin(entities), std::end(entities));
  1357. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1358. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1359. registry.compact<int>();
  1360. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1361. ASSERT_EQ(registry.storage<stable_type>().size(), 2u);
  1362. registry.compact();
  1363. ASSERT_EQ(registry.storage<int>().size(), 0u);
  1364. ASSERT_EQ(registry.storage<stable_type>().size(), 0u);
  1365. }
  1366. TEST(Registry, NonOwningGroupInterleaved) {
  1367. entt::registry registry;
  1368. typename entt::entity entity = entt::null;
  1369. entity = registry.create();
  1370. registry.emplace<int>(entity);
  1371. registry.emplace<char>(entity);
  1372. const auto group = registry.group(entt::get<int, char>);
  1373. entity = registry.create();
  1374. registry.emplace<int>(entity);
  1375. registry.emplace<char>(entity);
  1376. std::size_t cnt{};
  1377. group.each([&cnt](auto...) { ++cnt; });
  1378. ASSERT_EQ(cnt, 2u);
  1379. }
  1380. TEST(Registry, FullOwningGroupInterleaved) {
  1381. entt::registry registry;
  1382. typename entt::entity entity = entt::null;
  1383. entity = registry.create();
  1384. registry.emplace<int>(entity);
  1385. registry.emplace<char>(entity);
  1386. const auto group = registry.group<int, char>();
  1387. entity = registry.create();
  1388. registry.emplace<int>(entity);
  1389. registry.emplace<char>(entity);
  1390. std::size_t cnt{};
  1391. group.each([&cnt](auto...) { ++cnt; });
  1392. ASSERT_EQ(cnt, 2u);
  1393. }
  1394. TEST(Registry, PartialOwningGroupInterleaved) {
  1395. entt::registry registry;
  1396. typename entt::entity entity = entt::null;
  1397. entity = registry.create();
  1398. registry.emplace<int>(entity);
  1399. registry.emplace<char>(entity);
  1400. const auto group = registry.group<int>(entt::get<char>);
  1401. entity = registry.create();
  1402. registry.emplace<int>(entity);
  1403. registry.emplace<char>(entity);
  1404. std::size_t cnt{};
  1405. group.each([&cnt](auto...) { ++cnt; });
  1406. ASSERT_EQ(cnt, 2u);
  1407. }
  1408. TEST(Registry, NonOwningGroupSortInterleaved) {
  1409. entt::registry registry;
  1410. const auto group = registry.group(entt::get<int, char>);
  1411. const auto e0 = registry.create();
  1412. registry.emplace<int>(e0, 0);
  1413. registry.emplace<char>(e0, '0');
  1414. const auto e1 = registry.create();
  1415. registry.emplace<int>(e1, 1);
  1416. registry.emplace<char>(e1, '1');
  1417. registry.sort<int>(std::greater{});
  1418. registry.sort<char>(std::less{});
  1419. const auto e2 = registry.create();
  1420. registry.emplace<int>(e2, 2);
  1421. registry.emplace<char>(e2, '2');
  1422. group.each([e0, e1, e2](const auto entity, const auto &i, const auto &c) {
  1423. if(entity == e0) {
  1424. ASSERT_EQ(i, 0);
  1425. ASSERT_EQ(c, '0');
  1426. } else if(entity == e1) {
  1427. ASSERT_EQ(i, 1);
  1428. ASSERT_EQ(c, '1');
  1429. } else if(entity == e2) {
  1430. ASSERT_EQ(i, 2);
  1431. ASSERT_EQ(c, '2');
  1432. }
  1433. });
  1434. }
  1435. TEST(Registry, GetOrEmplace) {
  1436. entt::registry registry;
  1437. const auto entity = registry.create();
  1438. const auto value = registry.get_or_emplace<int>(entity, 3);
  1439. // get_or_emplace must work for empty types
  1440. static_cast<void>(registry.get_or_emplace<empty_type>(entity));
  1441. ASSERT_TRUE((registry.all_of<int, empty_type>(entity)));
  1442. ASSERT_EQ(registry.get<int>(entity), value);
  1443. ASSERT_EQ(registry.get<int>(entity), 3);
  1444. }
  1445. TEST(Registry, Constness) {
  1446. entt::registry registry;
  1447. static_assert((std::is_same_v<decltype(registry.emplace<int>({})), int &>));
  1448. static_assert((std::is_same_v<decltype(registry.emplace<empty_type>({})), void>));
  1449. static_assert((std::is_same_v<decltype(registry.get<>({})), std::tuple<>>));
  1450. static_assert((std::is_same_v<decltype(registry.get<int>({})), int &>));
  1451. static_assert((std::is_same_v<decltype(registry.get<int, const char>({})), std::tuple<int &, const char &>>));
  1452. static_assert((std::is_same_v<decltype(registry.try_get<>({})), std::tuple<>>));
  1453. static_assert((std::is_same_v<decltype(registry.try_get<int>({})), int *>));
  1454. static_assert((std::is_same_v<decltype(registry.try_get<int, const char>({})), std::tuple<int *, const char *>>));
  1455. static_assert((std::is_same_v<decltype(registry.ctx().get<int>()), int &>));
  1456. static_assert((std::is_same_v<decltype(registry.ctx().get<const char>()), const char &>));
  1457. static_assert((std::is_same_v<decltype(registry.ctx().find<int>()), int *>));
  1458. static_assert((std::is_same_v<decltype(registry.ctx().find<const char>()), const char *>));
  1459. static_assert((std::is_same_v<decltype(std::as_const(registry).get<>({})), std::tuple<>>));
  1460. static_assert((std::is_same_v<decltype(std::as_const(registry).get<int>({})), const int &>));
  1461. static_assert((std::is_same_v<decltype(std::as_const(registry).get<int, const char>({})), std::tuple<const int &, const char &>>));
  1462. static_assert((std::is_same_v<decltype(std::as_const(registry).try_get<>({})), std::tuple<>>));
  1463. static_assert((std::is_same_v<decltype(std::as_const(registry).try_get<int>({})), const int *>));
  1464. static_assert((std::is_same_v<decltype(std::as_const(registry).try_get<int, const char>({})), std::tuple<const int *, const char *>>));
  1465. static_assert((std::is_same_v<decltype(std::as_const(registry).ctx().get<int>()), const int &>));
  1466. static_assert((std::is_same_v<decltype(std::as_const(registry).ctx().get<const char>()), const char &>));
  1467. static_assert((std::is_same_v<decltype(std::as_const(registry).ctx().find<int>()), const int *>));
  1468. static_assert((std::is_same_v<decltype(std::as_const(registry).ctx().find<const char>()), const char *>));
  1469. }
  1470. TEST(Registry, MoveOnlyComponent) {
  1471. entt::registry registry;
  1472. // the purpose is to ensure that move only types are always accepted
  1473. registry.emplace<std::unique_ptr<int>>(registry.create());
  1474. }
  1475. TEST(Registry, NonDefaultConstructibleComponent) {
  1476. entt::registry registry;
  1477. // the purpose is to ensure that non default constructible type are always accepted
  1478. registry.emplace<non_default_constructible>(registry.create(), 42);
  1479. }
  1480. TEST(Registry, Dependencies) {
  1481. entt::registry registry;
  1482. const auto entity = registry.create();
  1483. // required because of an issue of VS2019
  1484. constexpr auto emplace_or_replace = &entt::registry::emplace_or_replace<double>;
  1485. constexpr auto remove = &entt::registry::remove<double>;
  1486. registry.on_construct<int>().connect<emplace_or_replace>();
  1487. registry.on_destroy<int>().connect<remove>();
  1488. registry.emplace<double>(entity, .3);
  1489. ASSERT_FALSE(registry.all_of<int>(entity));
  1490. ASSERT_EQ(registry.get<double>(entity), .3);
  1491. registry.emplace<int>(entity);
  1492. ASSERT_TRUE(registry.all_of<int>(entity));
  1493. ASSERT_EQ(registry.get<double>(entity), .0);
  1494. registry.erase<int>(entity);
  1495. ASSERT_FALSE((registry.any_of<int, double>(entity)));
  1496. registry.on_construct<int>().disconnect<emplace_or_replace>();
  1497. registry.on_destroy<int>().disconnect<remove>();
  1498. registry.emplace<int>(entity);
  1499. ASSERT_TRUE((registry.any_of<int, double>(entity)));
  1500. ASSERT_FALSE(registry.all_of<double>(entity));
  1501. }
  1502. TEST(Registry, StableEmplace) {
  1503. entt::registry registry;
  1504. registry.on_construct<int>().connect<&listener::sort<int>>();
  1505. registry.emplace<int>(registry.create(), 0);
  1506. ASSERT_EQ(registry.emplace<int>(registry.create(), 1), 1);
  1507. }
  1508. TEST(Registry, AssignEntities) {
  1509. using traits_type = entt::entt_traits<entt::entity>;
  1510. entt::registry registry;
  1511. entt::entity entities[3];
  1512. registry.create(std::begin(entities), std::end(entities));
  1513. registry.release(entities[1]);
  1514. registry.release(entities[2]);
  1515. entt::registry other;
  1516. const auto *data = registry.data();
  1517. other.assign(data, data + registry.storage<entt::entity>().size(), registry.released());
  1518. ASSERT_EQ(registry.storage<entt::entity>().size(), other.storage<entt::entity>().size());
  1519. ASSERT_TRUE(other.valid(entities[0]));
  1520. ASSERT_FALSE(other.valid(entities[1]));
  1521. ASSERT_FALSE(other.valid(entities[2]));
  1522. ASSERT_EQ(registry.create(), other.create());
  1523. ASSERT_EQ(traits_type::to_entity(other.create()), traits_type::to_integral(entities[1]));
  1524. }
  1525. TEST(Registry, ScramblingPoolsIsAllowed) {
  1526. entt::registry registry;
  1527. registry.on_destroy<int>().connect<&listener::sort<int>>();
  1528. for(std::size_t i{}; i < 2u; ++i) {
  1529. const auto entity = registry.create();
  1530. registry.emplace<int>(entity, static_cast<int>(i));
  1531. }
  1532. registry.destroy(registry.view<int>().back());
  1533. // thanks to @andranik3949 for pointing out this missing test
  1534. registry.view<const int>().each([](const auto entity, const auto &value) {
  1535. ASSERT_EQ(static_cast<int>(entt::to_integral(entity)), value);
  1536. });
  1537. }
  1538. TEST(Registry, RuntimePools) {
  1539. using namespace entt::literals;
  1540. entt::registry registry;
  1541. auto &storage = registry.storage<empty_type>("other"_hs);
  1542. const auto entity = registry.create();
  1543. static_assert(std::is_same_v<decltype(registry.storage<empty_type>()), entt::storage_type_t<empty_type> &>);
  1544. static_assert(std::is_same_v<decltype(std::as_const(registry).storage<empty_type>()), const entt::storage_type_t<empty_type> *>);
  1545. static_assert(std::is_same_v<decltype(registry.storage("other"_hs)), entt::storage_type_t<empty_type>::base_type *>);
  1546. static_assert(std::is_same_v<decltype(std::as_const(registry).storage("other"_hs)), const entt::storage_type_t<empty_type>::base_type *>);
  1547. ASSERT_NE(registry.storage("other"_hs), nullptr);
  1548. ASSERT_EQ(std::as_const(registry).storage("rehto"_hs), nullptr);
  1549. ASSERT_EQ(&registry.storage<empty_type>("other"_hs), &storage);
  1550. ASSERT_NE(std::as_const(registry).storage<empty_type>(), &storage);
  1551. ASSERT_FALSE(registry.any_of<empty_type>(entity));
  1552. ASSERT_FALSE(storage.contains(entity));
  1553. registry.emplace<empty_type>(entity);
  1554. ASSERT_FALSE(storage.contains(entity));
  1555. ASSERT_TRUE(registry.any_of<empty_type>(entity));
  1556. ASSERT_EQ((entt::basic_view{registry.storage<empty_type>(), storage}.size_hint()), 0u);
  1557. storage.emplace(entity);
  1558. ASSERT_TRUE(storage.contains(entity));
  1559. ASSERT_TRUE(registry.any_of<empty_type>(entity));
  1560. ASSERT_EQ((entt::basic_view{registry.storage<empty_type>(), storage}.size_hint()), 1u);
  1561. registry.destroy(entity);
  1562. ASSERT_EQ(registry.create(entity), entity);
  1563. ASSERT_FALSE(storage.contains(entity));
  1564. ASSERT_FALSE(registry.any_of<empty_type>(entity));
  1565. }
  1566. ENTT_DEBUG_TEST(RegistryDeathTest, RuntimePools) {
  1567. using namespace entt::literals;
  1568. entt::registry registry;
  1569. registry.storage<empty_type>("other"_hs);
  1570. ASSERT_DEATH(registry.storage<int>("other"_hs), "");
  1571. ASSERT_DEATH(std::as_const(registry).storage<int>("other"_hs), "");
  1572. }
  1573. TEST(Registry, Storage) {
  1574. using namespace entt::literals;
  1575. entt::registry registry;
  1576. const auto entity = registry.create();
  1577. auto &storage = registry.storage<int>("int"_hs);
  1578. storage.emplace(entity);
  1579. for(auto [id, pool]: registry.storage()) {
  1580. static_assert(std::is_same_v<decltype(pool), entt::sparse_set &>);
  1581. static_assert(std::is_same_v<decltype(id), entt::id_type>);
  1582. ASSERT_TRUE(pool.contains(entity));
  1583. ASSERT_EQ(std::addressof(storage), std::addressof(pool));
  1584. ASSERT_EQ(id, "int"_hs);
  1585. }
  1586. for(auto &&curr: std::as_const(registry).storage()) {
  1587. static_assert(std::is_same_v<decltype(curr.second), const entt::sparse_set &>);
  1588. static_assert(std::is_same_v<decltype(curr.first), entt::id_type>);
  1589. ASSERT_TRUE(curr.second.contains(entity));
  1590. ASSERT_EQ(std::addressof(storage), std::addressof(curr.second));
  1591. ASSERT_EQ(curr.first, "int"_hs);
  1592. }
  1593. }
  1594. TEST(Registry, RegistryStorageIterator) {
  1595. entt::registry registry;
  1596. const auto entity = registry.create();
  1597. registry.emplace<int>(entity);
  1598. auto test = [entity](auto iterable) {
  1599. auto end{iterable.begin()};
  1600. decltype(end) begin{};
  1601. begin = iterable.end();
  1602. std::swap(begin, end);
  1603. ASSERT_EQ(begin, iterable.cbegin());
  1604. ASSERT_EQ(end, iterable.cend());
  1605. ASSERT_NE(begin, end);
  1606. ASSERT_EQ(begin++, iterable.begin());
  1607. ASSERT_EQ(begin--, iterable.end());
  1608. ASSERT_EQ(begin + 1, iterable.end());
  1609. ASSERT_EQ(end - 1, iterable.begin());
  1610. ASSERT_EQ(++begin, iterable.end());
  1611. ASSERT_EQ(--begin, iterable.begin());
  1612. ASSERT_EQ(begin += 1, iterable.end());
  1613. ASSERT_EQ(begin -= 1, iterable.begin());
  1614. ASSERT_EQ(begin + (end - begin), iterable.end());
  1615. ASSERT_EQ(begin - (begin - end), iterable.end());
  1616. ASSERT_EQ(end - (end - begin), iterable.begin());
  1617. ASSERT_EQ(end + (begin - end), iterable.begin());
  1618. ASSERT_EQ(begin[0u].first, iterable.begin()->first);
  1619. ASSERT_EQ(std::addressof(begin[0u].second), std::addressof((*iterable.begin()).second));
  1620. ASSERT_LT(begin, end);
  1621. ASSERT_LE(begin, iterable.begin());
  1622. ASSERT_GT(end, begin);
  1623. ASSERT_GE(end, iterable.end());
  1624. ASSERT_EQ(begin[0u].first, entt::type_id<int>().hash());
  1625. ASSERT_TRUE(begin[0u].second.contains(entity));
  1626. };
  1627. test(registry.storage());
  1628. test(std::as_const(registry).storage());
  1629. decltype(std::as_const(registry).storage().begin()) cit = registry.storage().begin();
  1630. ASSERT_EQ(cit, registry.storage().begin());
  1631. ASSERT_NE(cit, std::as_const(registry).storage().end());
  1632. }
  1633. TEST(Registry, RegistryStorageIteratorConversion) {
  1634. entt::registry registry;
  1635. registry.storage<int>();
  1636. auto proxy = registry.storage();
  1637. auto cproxy = std::as_const(registry).storage();
  1638. typename decltype(proxy)::iterator it = proxy.begin();
  1639. typename decltype(cproxy)::iterator cit = it;
  1640. static_assert(std::is_same_v<decltype(*it), std::pair<entt::id_type, entt::sparse_set &>>);
  1641. static_assert(std::is_same_v<decltype(*cit), std::pair<entt::id_type, const entt::sparse_set &>>);
  1642. ASSERT_EQ(it->first, entt::type_id<int>().hash());
  1643. ASSERT_EQ((*it).second.type(), entt::type_id<int>());
  1644. ASSERT_EQ(it->first, cit->first);
  1645. ASSERT_EQ((*it).second.type(), (*cit).second.type());
  1646. ASSERT_EQ(it - cit, 0);
  1647. ASSERT_EQ(cit - it, 0);
  1648. ASSERT_LE(it, cit);
  1649. ASSERT_LE(cit, it);
  1650. ASSERT_GE(it, cit);
  1651. ASSERT_GE(cit, it);
  1652. ASSERT_EQ(it, cit);
  1653. ASSERT_NE(++cit, it);
  1654. }
  1655. TEST(Registry, VoidType) {
  1656. using namespace entt::literals;
  1657. entt::registry registry;
  1658. const auto entity = registry.create();
  1659. auto &storage = registry.storage<void>("custom"_hs);
  1660. storage.emplace(entity);
  1661. ASSERT_TRUE(registry.storage<void>().empty());
  1662. ASSERT_FALSE(registry.storage<void>("custom"_hs).empty());
  1663. ASSERT_TRUE(registry.storage<void>("custom"_hs).contains(entity));
  1664. ASSERT_EQ(registry.storage<void>().type(), entt::type_id<void>());
  1665. ASSERT_EQ(registry.storage<void>("custom"_hs).type(), entt::type_id<void>());
  1666. }
  1667. TEST(Registry, NoEtoType) {
  1668. entt::registry registry;
  1669. const auto entity = registry.create();
  1670. registry.emplace<no_eto_type>(entity);
  1671. registry.emplace<int>(entity, 42);
  1672. ASSERT_NE(registry.storage<no_eto_type>().raw(), nullptr);
  1673. ASSERT_NE(registry.try_get<no_eto_type>(entity), nullptr);
  1674. ASSERT_EQ(registry.view<no_eto_type>().get(entity), std::as_const(registry).view<const no_eto_type>().get(entity));
  1675. auto view = registry.view<no_eto_type, int>();
  1676. auto cview = std::as_const(registry).view<const no_eto_type, const int>();
  1677. ASSERT_EQ((std::get<0>(view.get<no_eto_type, int>(entity))), (std::get<0>(cview.get<const no_eto_type, const int>(entity))));
  1678. }
  1679. TEST(Registry, CtxAndPoolMemberDestructionOrder) {
  1680. auto registry = std::make_unique<entt::registry>();
  1681. const auto entity = registry->create();
  1682. bool ctx_check = false;
  1683. registry->ctx().emplace<typename destruction_order::ctx_check_type>();
  1684. registry->emplace<destruction_order>(entity, *registry, ctx_check);
  1685. registry.reset();
  1686. ASSERT_TRUE(ctx_check);
  1687. }