registry.cpp 71 KB

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