registry.cpp 73 KB

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