registry.cpp 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362
  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<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, ComponentsWithTypesFromStandardTemplateLibrary) {
  570. // see #37 - the test shouldn't crash, that's all
  571. entt::registry registry;
  572. const auto entity = registry.create();
  573. registry.assign<std::unordered_set<int>>(entity).insert(42);
  574. registry.destroy(entity);
  575. }
  576. TEST(Registry, ConstructWithComponents) {
  577. // it should compile, that's all
  578. entt::registry registry;
  579. const auto value = 0;
  580. registry.assign<int>(registry.create(), value);
  581. }
  582. TEST(Registry, MergeTwoRegistries) {
  583. entt::registry src;
  584. entt::registry dst;
  585. std::unordered_map<entt::entity, entt::entity> ref;
  586. auto merge = [&ref, &dst](const auto &view) {
  587. view.each([&](auto entity, const auto &component) {
  588. if(ref.find(entity) == ref.cend()) {
  589. const auto other = dst.create();
  590. dst.template assign<std::decay_t<decltype(component)>>(other, component);
  591. ref.emplace(entity, other);
  592. } else {
  593. using component_type = std::decay_t<decltype(component)>;
  594. dst.template assign<component_type>(ref[entity], component);
  595. }
  596. });
  597. };
  598. auto e0 = src.create();
  599. src.assign<int>(e0);
  600. src.assign<float>(e0);
  601. src.assign<double>(e0);
  602. auto e1 = src.create();
  603. src.assign<char>(e1);
  604. src.assign<float>(e1);
  605. src.assign<int>(e1);
  606. auto e2 = dst.create();
  607. dst.assign<int>(e2);
  608. dst.assign<char>(e2);
  609. dst.assign<double>(e2);
  610. auto e3 = dst.create();
  611. dst.assign<float>(e3);
  612. dst.assign<int>(e3);
  613. auto eq = [](auto begin, auto end) { ASSERT_EQ(begin, end); };
  614. auto ne = [](auto begin, auto end) { ASSERT_NE(begin, end); };
  615. eq(dst.view<int, float, double>().begin(), dst.view<int, float, double>().end());
  616. eq(dst.view<char, float, int>().begin(), dst.view<char, float, int>().end());
  617. merge(src.view<int>());
  618. merge(src.view<char>());
  619. merge(src.view<double>());
  620. merge(src.view<float>());
  621. ne(dst.view<int, float, double>().begin(), dst.view<int, float, double>().end());
  622. ne(dst.view<char, float, int>().begin(), dst.view<char, float, int>().end());
  623. }
  624. TEST(Registry, Signals) {
  625. entt::registry registry;
  626. listener listener;
  627. registry.on_construct<empty_type>().connect<&listener::incr<empty_type>>(&listener);
  628. registry.on_destroy<empty_type>().connect<&listener::decr<empty_type>>(&listener);
  629. registry.on_construct<int>().connect<&listener::incr<int>>(&listener);
  630. registry.on_destroy<int>().connect<&listener::decr<int>>(&listener);
  631. auto e0 = registry.create();
  632. auto e1 = registry.create();
  633. registry.assign<empty_type>(e0);
  634. registry.assign<empty_type>(e1);
  635. ASSERT_EQ(listener.counter, 2);
  636. ASSERT_EQ(listener.last, e1);
  637. registry.assign<int>(e1);
  638. registry.assign<int>(e0);
  639. ASSERT_EQ(listener.counter, 4);
  640. ASSERT_EQ(listener.last, e0);
  641. registry.remove<empty_type>(e0);
  642. registry.remove<int>(e0);
  643. ASSERT_EQ(listener.counter, 2);
  644. ASSERT_EQ(listener.last, e0);
  645. registry.on_destroy<empty_type>().disconnect<&listener::decr<empty_type>>(&listener);
  646. registry.on_destroy<int>().disconnect<&listener::decr<int>>(&listener);
  647. registry.remove<empty_type>(e1);
  648. registry.remove<int>(e1);
  649. ASSERT_EQ(listener.counter, 2);
  650. ASSERT_EQ(listener.last, e0);
  651. registry.on_construct<empty_type>().disconnect<&listener::incr<empty_type>>(&listener);
  652. registry.on_construct<int>().disconnect<&listener::incr<int>>(&listener);
  653. registry.assign<empty_type>(e1);
  654. registry.assign<int>(e1);
  655. ASSERT_EQ(listener.counter, 2);
  656. ASSERT_EQ(listener.last, e0);
  657. registry.on_construct<int>().connect<&listener::incr<int>>(&listener);
  658. registry.on_destroy<int>().connect<&listener::decr<int>>(&listener);
  659. registry.assign<int>(e0);
  660. registry.reset<int>(e1);
  661. ASSERT_EQ(listener.counter, 2);
  662. ASSERT_EQ(listener.last, e1);
  663. registry.on_construct<empty_type>().connect<&listener::incr<empty_type>>(&listener);
  664. registry.on_destroy<empty_type>().connect<&listener::decr<empty_type>>(&listener);
  665. registry.reset<empty_type>(e1);
  666. registry.assign<empty_type>(e0);
  667. ASSERT_EQ(listener.counter, 2);
  668. ASSERT_EQ(listener.last, e0);
  669. registry.reset<empty_type>();
  670. registry.reset<int>();
  671. ASSERT_EQ(listener.counter, 0);
  672. ASSERT_EQ(listener.last, e0);
  673. registry.assign<empty_type>(e0);
  674. registry.assign<empty_type>(e1);
  675. registry.assign<int>(e0);
  676. registry.assign<int>(e1);
  677. registry.destroy(e1);
  678. ASSERT_EQ(listener.counter, 2);
  679. ASSERT_EQ(listener.last, e1);
  680. registry.remove<int>(e0);
  681. registry.remove<empty_type>(e0);
  682. registry.assign_or_replace<int>(e0);
  683. registry.assign_or_replace<empty_type>(e0);
  684. ASSERT_EQ(listener.counter, 2);
  685. ASSERT_EQ(listener.last, e0);
  686. registry.on_destroy<empty_type>().disconnect<&listener::decr<empty_type>>(&listener);
  687. registry.on_destroy<int>().disconnect<&listener::decr<int>>(&listener);
  688. registry.assign_or_replace<empty_type>(e0);
  689. registry.assign_or_replace<int>(e0);
  690. ASSERT_EQ(listener.counter, 2);
  691. ASSERT_EQ(listener.last, e0);
  692. registry.on_replace<empty_type>().connect<&listener::incr<empty_type>>(&listener);
  693. registry.on_replace<int>().connect<&listener::incr<int>>(&listener);
  694. registry.assign_or_replace<empty_type>(e0);
  695. registry.assign_or_replace<int>(e0);
  696. ASSERT_EQ(listener.counter, 4);
  697. ASSERT_EQ(listener.last, e0);
  698. registry.replace<empty_type>(e0);
  699. registry.replace<int>(e0);
  700. ASSERT_EQ(listener.counter, 6);
  701. ASSERT_EQ(listener.last, e0);
  702. }
  703. TEST(Registry, DestroyByComponents) {
  704. entt::registry registry;
  705. const auto e0 = registry.create();
  706. const auto e1 = registry.create();
  707. const auto e2 = registry.create();
  708. registry.assign<int>(e0);
  709. registry.assign<char>(e0);
  710. registry.assign<double>(e0);
  711. registry.assign<int>(e1);
  712. registry.assign<char>(e1);
  713. registry.assign<int>(e2);
  714. ASSERT_TRUE(registry.valid(e0));
  715. ASSERT_TRUE(registry.valid(e1));
  716. ASSERT_TRUE(registry.valid(e2));
  717. {
  718. const auto view = registry.view<int, char>();
  719. registry.destroy(view.begin(), view.end());
  720. }
  721. ASSERT_FALSE(registry.valid(e0));
  722. ASSERT_FALSE(registry.valid(e1));
  723. ASSERT_TRUE(registry.valid(e2));
  724. {
  725. const auto view = registry.view<int>();
  726. registry.destroy(view.begin(), view.end());
  727. }
  728. ASSERT_FALSE(registry.valid(e0));
  729. ASSERT_FALSE(registry.valid(e1));
  730. ASSERT_FALSE(registry.valid(e2));
  731. }
  732. TEST(Registry, CreateManyEntitiesAtOnce) {
  733. entt::registry registry;
  734. entt::entity entities[3];
  735. const auto entity = registry.create();
  736. registry.destroy(registry.create());
  737. registry.destroy(entity);
  738. registry.destroy(registry.create());
  739. registry.create(std::begin(entities), std::end(entities));
  740. ASSERT_TRUE(registry.valid(entities[0]));
  741. ASSERT_TRUE(registry.valid(entities[1]));
  742. ASSERT_TRUE(registry.valid(entities[2]));
  743. ASSERT_EQ(registry.entity(entities[0]), entt::entity{0});
  744. ASSERT_EQ(registry.version(entities[0]), entt::registry::version_type{2});
  745. ASSERT_EQ(registry.entity(entities[1]), entt::entity{1});
  746. ASSERT_EQ(registry.version(entities[1]), entt::registry::version_type{1});
  747. ASSERT_EQ(registry.entity(entities[2]), entt::entity{2});
  748. ASSERT_EQ(registry.version(entities[2]), entt::registry::version_type{0});
  749. }
  750. TEST(Registry, CreateAnEntityWithComponents) {
  751. entt::registry registry;
  752. auto &&[entity, ivalue, cvalue, evalue] = registry.create<int, char, empty_type>();
  753. // suppress warnings
  754. (void)evalue;
  755. ASSERT_FALSE(registry.empty<int>());
  756. ASSERT_FALSE(registry.empty<char>());
  757. ASSERT_FALSE(registry.empty<empty_type>());
  758. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{1});
  759. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{1});
  760. ASSERT_EQ(registry.size<empty_type>(), entt::registry::size_type{1});
  761. ASSERT_TRUE((registry.has<int, char, empty_type>(entity)));
  762. ivalue = 42;
  763. cvalue = 'c';
  764. ASSERT_EQ(registry.get<int>(entity), 42);
  765. ASSERT_EQ(registry.get<char>(entity), 'c');
  766. }
  767. TEST(Registry, CreateManyEntitiesWithComponentsAtOnce) {
  768. entt::registry registry;
  769. entt::entity entities[3];
  770. const auto entity = registry.create();
  771. registry.destroy(registry.create());
  772. registry.destroy(entity);
  773. registry.destroy(registry.create());
  774. const auto [iptr, cptr, eptr] = registry.create<int, char, empty_type>(std::begin(entities), std::end(entities));
  775. ASSERT_FALSE(registry.empty<int>());
  776. ASSERT_FALSE(registry.empty<char>());
  777. ASSERT_FALSE(registry.empty<empty_type>());
  778. ASSERT_NE(iptr, nullptr);
  779. ASSERT_NE(cptr, nullptr);
  780. ASSERT_EQ(eptr, nullptr);
  781. ASSERT_EQ(registry.size<int>(), entt::registry::size_type{3});
  782. ASSERT_EQ(registry.size<char>(), entt::registry::size_type{3});
  783. ASSERT_EQ(registry.size<empty_type>(), entt::registry::size_type{3});
  784. ASSERT_TRUE(registry.valid(entities[0]));
  785. ASSERT_TRUE(registry.valid(entities[1]));
  786. ASSERT_TRUE(registry.valid(entities[2]));
  787. ASSERT_EQ(registry.entity(entities[0]), entt::entity{0});
  788. ASSERT_EQ(registry.version(entities[0]), entt::registry::version_type{2});
  789. ASSERT_EQ(registry.entity(entities[1]), entt::entity{1});
  790. ASSERT_EQ(registry.version(entities[1]), entt::registry::version_type{1});
  791. ASSERT_EQ(registry.entity(entities[2]), entt::entity{2});
  792. ASSERT_EQ(registry.version(entities[2]), entt::registry::version_type{0});
  793. ASSERT_TRUE((registry.has<int, char, empty_type>(entities[0])));
  794. ASSERT_TRUE((registry.has<int, char, empty_type>(entities[1])));
  795. ASSERT_TRUE((registry.has<int, char, empty_type>(entities[2])));
  796. for(auto i = 0; i < 3; ++i) {
  797. iptr[i] = i;
  798. cptr[i] = char('a'+i);
  799. }
  800. for(auto i = 0; i < 3; ++i) {
  801. ASSERT_EQ(registry.get<int>(entities[i]), i);
  802. ASSERT_EQ(registry.get<char>(entities[i]), char('a'+i));
  803. }
  804. }
  805. TEST(Registry, CreateManyEntitiesWithComponentsAtOnceWithListener) {
  806. entt::registry registry;
  807. entt::entity entities[3];
  808. listener listener;
  809. registry.on_construct<int>().connect<&listener::incr<int>>(&listener);
  810. registry.create<int, char>(std::begin(entities), std::end(entities));
  811. ASSERT_EQ(listener.counter, 3);
  812. registry.on_construct<int>().disconnect<&listener::incr<int>>(&listener);
  813. registry.on_construct<empty_type>().connect<&listener::incr<empty_type>>(&listener);
  814. registry.create<char, empty_type>(std::begin(entities), std::end(entities));
  815. ASSERT_EQ(listener.counter, 6);
  816. }
  817. TEST(Registry, NonOwningGroupInterleaved) {
  818. entt::registry registry;
  819. typename entt::entity entity = entt::null;
  820. entity = registry.create();
  821. registry.assign<int>(entity);
  822. registry.assign<char>(entity);
  823. const auto group = registry.group<>(entt::get<int, char>);
  824. entity = registry.create();
  825. registry.assign<int>(entity);
  826. registry.assign<char>(entity);
  827. decltype(group)::size_type cnt{0};
  828. group.each([&cnt](auto...) { ++cnt; });
  829. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  830. }
  831. TEST(Registry, FullOwningGroupInterleaved) {
  832. entt::registry registry;
  833. typename entt::entity entity = entt::null;
  834. entity = registry.create();
  835. registry.assign<int>(entity);
  836. registry.assign<char>(entity);
  837. const auto group = registry.group<int, char>();
  838. entity = registry.create();
  839. registry.assign<int>(entity);
  840. registry.assign<char>(entity);
  841. decltype(group)::size_type cnt{0};
  842. group.each([&cnt](auto...) { ++cnt; });
  843. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  844. }
  845. TEST(Registry, PartialOwningGroupInterleaved) {
  846. entt::registry registry;
  847. typename entt::entity entity = entt::null;
  848. entity = registry.create();
  849. registry.assign<int>(entity);
  850. registry.assign<char>(entity);
  851. const auto group = registry.group<int>(entt::get<char>);
  852. entity = registry.create();
  853. registry.assign<int>(entity);
  854. registry.assign<char>(entity);
  855. decltype(group)::size_type cnt{0};
  856. group.each([&cnt](auto...) { ++cnt; });
  857. ASSERT_EQ(cnt, decltype(group)::size_type{2});
  858. }
  859. TEST(Registry, NonOwningGroupSortInterleaved) {
  860. entt::registry registry;
  861. const auto group = registry.group<>(entt::get<int, char>);
  862. const auto e0 = registry.create();
  863. registry.assign<int>(e0, 0);
  864. registry.assign<char>(e0, '0');
  865. const auto e1 = registry.create();
  866. registry.assign<int>(e1, 1);
  867. registry.assign<char>(e1, '1');
  868. registry.sort<int>([](auto lhs, auto rhs) { return lhs > rhs; });
  869. registry.sort<char>([](auto lhs, auto rhs) { return lhs < rhs; });
  870. const auto e2 = registry.create();
  871. registry.assign<int>(e2, 2);
  872. registry.assign<char>(e2, '2');
  873. group.each([e0, e1, e2](const auto entity, const auto &i, const auto &c) {
  874. if(entity == e0) {
  875. ASSERT_EQ(i, 0);
  876. ASSERT_EQ(c, '0');
  877. } else if(entity == e1) {
  878. ASSERT_EQ(i, 1);
  879. ASSERT_EQ(c, '1');
  880. } else if(entity == e2) {
  881. ASSERT_EQ(i, 2);
  882. ASSERT_EQ(c, '2');
  883. }
  884. });
  885. }
  886. TEST(Registry, Clone) {
  887. entt::registry registry;
  888. entt::registry other;
  889. registry.destroy(registry.create());
  890. const auto e0 = registry.create();
  891. registry.assign<int>(e0, 0);
  892. registry.assign<double>(e0, 0.0);
  893. const auto e1 = registry.create();
  894. registry.assign<int>(e1, 1);
  895. registry.assign<char>(e1, '1');
  896. registry.assign<double>(e1, 1.1);
  897. const auto e2 = registry.create();
  898. registry.assign<int>(e2, 2);
  899. registry.assign<char>(e2, '2');
  900. registry.destroy(e1);
  901. ASSERT_EQ((other.group<int, char>().size()), entt::registry::size_type{0});
  902. other = registry.clone<int, char>();
  903. ASSERT_EQ((other.group<int, char>().size()), entt::registry::size_type{1});
  904. ASSERT_EQ(other.size(), registry.size());
  905. ASSERT_EQ(other.alive(), registry.alive());
  906. ASSERT_TRUE(other.valid(e0));
  907. ASSERT_FALSE(other.valid(e1));
  908. ASSERT_TRUE(other.valid(e2));
  909. ASSERT_TRUE((other.has<int>(e0)));
  910. ASSERT_FALSE((other.has<double>(e0)));
  911. ASSERT_TRUE((other.has<int, char>(e2)));
  912. ASSERT_EQ(other.get<int>(e0), 0);
  913. ASSERT_EQ(other.get<int>(e2), 2);
  914. ASSERT_EQ(other.get<char>(e2), '2');
  915. const auto e3 = other.create();
  916. ASSERT_NE(e1, e3);
  917. ASSERT_EQ(registry.entity(e1), registry.entity(e3));
  918. ASSERT_EQ(other.entity(e1), other.entity(e3));
  919. other.assign<int>(e3, 3);
  920. other.assign<char>(e3, '3');
  921. ASSERT_EQ((registry.group<int, char>().size()), entt::registry::size_type{1});
  922. ASSERT_EQ((other.group<int, char>().size()), entt::registry::size_type{2});
  923. other = registry.clone();
  924. ASSERT_EQ(other.size(), registry.size());
  925. ASSERT_EQ(other.alive(), registry.alive());
  926. ASSERT_TRUE(other.valid(e0));
  927. ASSERT_FALSE(other.valid(e1));
  928. ASSERT_TRUE(other.valid(e2));
  929. ASSERT_FALSE(other.valid(e3));
  930. ASSERT_TRUE((other.has<int, double>(e0)));
  931. ASSERT_TRUE((other.has<int, char>(e2)));
  932. ASSERT_EQ(other.get<int>(e0), 0);
  933. ASSERT_EQ(other.get<double>(e0), 0.);
  934. ASSERT_EQ(other.get<int>(e2), 2);
  935. ASSERT_EQ(other.get<char>(e2), '2');
  936. other = other.clone<char>();
  937. ASSERT_EQ(other.size(), registry.size());
  938. ASSERT_EQ(other.alive(), registry.alive());
  939. ASSERT_TRUE(other.valid(e0));
  940. ASSERT_FALSE(other.valid(e1));
  941. ASSERT_TRUE(other.valid(e2));
  942. ASSERT_FALSE(other.valid(e3));
  943. ASSERT_FALSE((other.has<int>(e0)));
  944. ASSERT_FALSE((other.has<double>(e0)));
  945. ASSERT_FALSE((other.has<int>(e2)));
  946. ASSERT_TRUE((other.has<char>(e2)));
  947. ASSERT_TRUE(other.orphan(e0));
  948. ASSERT_EQ(other.get<char>(e2), '2');
  949. }
  950. TEST(Registry, CloneMoveOnlyComponent) {
  951. entt::registry registry;
  952. const auto entity = registry.create();
  953. registry.assign<std::unique_ptr<int>>(entity);
  954. registry.assign<char>(entity);
  955. auto other = registry.clone();
  956. ASSERT_TRUE(other.valid(entity));
  957. ASSERT_TRUE(other.has<char>(entity));
  958. ASSERT_FALSE(other.has<std::unique_ptr<int>>(entity));
  959. }
  960. TEST(Registry, GetOrAssign) {
  961. entt::registry registry;
  962. const auto entity = registry.create();
  963. const auto value = registry.get_or_assign<int>(entity, 3);
  964. ASSERT_TRUE(registry.has<int>(entity));
  965. ASSERT_EQ(registry.get<int>(entity), value);
  966. ASSERT_EQ(registry.get<int>(entity), 3);
  967. }
  968. TEST(Registry, Constness) {
  969. entt::registry registry;
  970. ASSERT_TRUE((std::is_same_v<decltype(registry.get<int>({})), int &>));
  971. ASSERT_TRUE((std::is_same_v<decltype(registry.get<int, char>({})), std::tuple<int &, char &>>));
  972. ASSERT_TRUE((std::is_same_v<decltype(registry.try_get<int>({})), int *>));
  973. ASSERT_TRUE((std::is_same_v<decltype(registry.try_get<int, char>({})), std::tuple<int *, char *>>));
  974. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).get<int>({})), const int &>));
  975. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).get<int, char>({})), std::tuple<const int &, const char &>>));
  976. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).try_get<int>({})), const int *>));
  977. ASSERT_TRUE((std::is_same_v<decltype(std::as_const(registry).try_get<int, char>({})), std::tuple<const int *, const char *>>));
  978. }
  979. TEST(Registry, BatchCreateAmbiguousCall) {
  980. struct ambiguous { std::uint32_t foo; std::uint64_t bar; };
  981. entt::registry registry;
  982. const auto entity = registry.create();
  983. std::uint32_t foo = 32u;
  984. std::uint64_t bar = 64u;
  985. // this should work, no other tests required
  986. registry.assign<ambiguous>(entity, foo, bar);
  987. }
  988. TEST(Registry, MoveOnlyComponent) {
  989. // the purpose is to ensure that move only components are always accepted
  990. entt::registry registry;
  991. const auto entity = registry.create();
  992. registry.assign<std::unique_ptr<int>>(entity);
  993. }