registry.cpp 42 KB

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