registry.cpp 65 KB

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