group.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685
  1. #include <utility>
  2. #include <iterator>
  3. #include <algorithm>
  4. #include <gtest/gtest.h>
  5. #include <entt/entity/registry.hpp>
  6. #include <entt/entity/group.hpp>
  7. TEST(NonOwningGroup, Functionalities) {
  8. entt::registry<> registry;
  9. auto group = registry.group<>(entt::get<int, char>);
  10. auto cgroup = std::as_const(registry).group<>(entt::get<const int, const char>);
  11. ASSERT_TRUE(group.empty());
  12. const auto e0 = registry.create();
  13. registry.assign<char>(e0);
  14. const auto e1 = registry.create();
  15. registry.assign<int>(e1);
  16. registry.assign<char>(e1);
  17. ASSERT_FALSE(group.empty());
  18. ASSERT_NO_THROW((group.begin()++));
  19. ASSERT_NO_THROW((++cgroup.begin()));
  20. ASSERT_NE(group.begin(), group.end());
  21. ASSERT_NE(cgroup.begin(), cgroup.end());
  22. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  23. registry.assign<int>(e0);
  24. ASSERT_EQ(group.size(), typename decltype(group)::size_type{2});
  25. registry.remove<int>(e0);
  26. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  27. registry.get<char>(e0) = '1';
  28. registry.get<char>(e1) = '2';
  29. registry.get<int>(e1) = 42;
  30. for(auto entity: group) {
  31. ASSERT_EQ(std::get<0>(cgroup.get<const int, const char>(entity)), 42);
  32. ASSERT_EQ(std::get<1>(group.get<int, char>(entity)), '2');
  33. ASSERT_EQ(cgroup.get<const char>(entity), '2');
  34. }
  35. ASSERT_EQ(*(group.data() + 0), e1);
  36. registry.remove<char>(e0);
  37. registry.remove<char>(e1);
  38. ASSERT_EQ(group.begin(), group.end());
  39. ASSERT_EQ(cgroup.begin(), cgroup.end());
  40. ASSERT_TRUE(group.empty());
  41. }
  42. TEST(OwningGroup, Functionalities) {
  43. entt::registry<> registry;
  44. auto group = registry.group<int>(entt::get<char>);
  45. auto cgroup = std::as_const(registry).group<const int>(entt::get<const char>);
  46. ASSERT_TRUE(group.empty());
  47. const auto e0 = registry.create();
  48. registry.assign<char>(e0);
  49. const auto e1 = registry.create();
  50. registry.assign<int>(e1);
  51. registry.assign<char>(e1);
  52. ASSERT_FALSE(group.empty());
  53. ASSERT_NO_THROW((group.begin()++));
  54. ASSERT_NO_THROW((++cgroup.begin()));
  55. ASSERT_NE(group.begin(), group.end());
  56. ASSERT_NE(cgroup.begin(), cgroup.end());
  57. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  58. registry.assign<int>(e0);
  59. ASSERT_EQ(group.size(), typename decltype(group)::size_type{2});
  60. registry.remove<int>(e0);
  61. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  62. registry.get<char>(e0) = '1';
  63. registry.get<char>(e1) = '2';
  64. registry.get<int>(e1) = 42;
  65. ASSERT_EQ(*(cgroup.raw<const int>() + 0), 42);
  66. ASSERT_EQ(*(group.raw<int>() + 0), 42);
  67. for(auto entity: group) {
  68. ASSERT_EQ(std::get<0>(cgroup.get<const int, const char>(entity)), 42);
  69. ASSERT_EQ(std::get<1>(group.get<int, char>(entity)), '2');
  70. ASSERT_EQ(cgroup.get<const char>(entity), '2');
  71. }
  72. ASSERT_EQ(*(group.data() + 0), e1);
  73. registry.remove<char>(e0);
  74. registry.remove<char>(e1);
  75. ASSERT_EQ(group.begin(), group.end());
  76. ASSERT_EQ(cgroup.begin(), cgroup.end());
  77. ASSERT_TRUE(group.empty());
  78. }
  79. TEST(NonOwningGroup, ElementAccess) {
  80. entt::registry<> registry;
  81. auto group = registry.group<>(entt::get<int, char>);
  82. auto cgroup = std::as_const(registry).group<>(entt::get<const int, const char>);
  83. const auto e0 = registry.create();
  84. registry.assign<int>(e0);
  85. registry.assign<char>(e0);
  86. const auto e1 = registry.create();
  87. registry.assign<int>(e1);
  88. registry.assign<char>(e1);
  89. for(typename decltype(group)::size_type i{}; i < group.size(); ++i) {
  90. ASSERT_EQ(group[i], i ? e0 : e1);
  91. ASSERT_EQ(cgroup[i], i ? e0 : e1);
  92. }
  93. }
  94. TEST(OwningGroup, ElementAccess) {
  95. entt::registry<> registry;
  96. auto group = registry.group<int>(entt::get<char>);
  97. auto cgroup = std::as_const(registry).group<const int>(entt::get<const char>);
  98. const auto e0 = registry.create();
  99. registry.assign<int>(e0);
  100. registry.assign<char>(e0);
  101. const auto e1 = registry.create();
  102. registry.assign<int>(e1);
  103. registry.assign<char>(e1);
  104. for(typename decltype(group)::size_type i{}; i < group.size(); ++i) {
  105. ASSERT_EQ(group[i], i ? e0 : e1);
  106. ASSERT_EQ(cgroup[i], i ? e0 : e1);
  107. }
  108. }
  109. TEST(NonOwningGroup, Contains) {
  110. entt::registry<> registry;
  111. auto group = registry.group<>(entt::get<int, char>);
  112. const auto e0 = registry.create();
  113. registry.assign<int>(e0);
  114. registry.assign<char>(e0);
  115. const auto e1 = registry.create();
  116. registry.assign<int>(e1);
  117. registry.assign<char>(e1);
  118. registry.destroy(e0);
  119. ASSERT_FALSE(group.contains(e0));
  120. ASSERT_TRUE(group.contains(e1));
  121. }
  122. TEST(OwningGroup, Contains) {
  123. entt::registry<> registry;
  124. auto group = registry.group<int>(entt::get<char>);
  125. const auto e0 = registry.create();
  126. registry.assign<int>(e0);
  127. registry.assign<char>(e0);
  128. const auto e1 = registry.create();
  129. registry.assign<int>(e1);
  130. registry.assign<char>(e1);
  131. registry.destroy(e0);
  132. ASSERT_FALSE(group.contains(e0));
  133. ASSERT_TRUE(group.contains(e1));
  134. }
  135. TEST(NonOwningGroup, Empty) {
  136. entt::registry<> registry;
  137. const auto e0 = registry.create();
  138. registry.assign<double>(e0);
  139. registry.assign<int>(e0);
  140. registry.assign<float>(e0);
  141. const auto e1 = registry.create();
  142. registry.assign<char>(e1);
  143. registry.assign<float>(e1);
  144. for(auto entity: registry.group<>(entt::get<char, int, float>)) {
  145. (void)entity;
  146. FAIL();
  147. }
  148. for(auto entity: registry.group<>(entt::get<double, char, int, float>)) {
  149. (void)entity;
  150. FAIL();
  151. }
  152. }
  153. TEST(OwningGroup, Empty) {
  154. entt::registry<> registry;
  155. const auto e0 = registry.create();
  156. registry.assign<double>(e0);
  157. registry.assign<int>(e0);
  158. registry.assign<float>(e0);
  159. const auto e1 = registry.create();
  160. registry.assign<char>(e1);
  161. registry.assign<float>(e1);
  162. for(auto entity: registry.group<char, int>(entt::get<float>)) {
  163. (void)entity;
  164. FAIL();
  165. }
  166. for(auto entity: registry.group<char, int>(entt::get<double, float>)) {
  167. (void)entity;
  168. FAIL();
  169. }
  170. }
  171. TEST(NonOwningGroup, Each) {
  172. entt::registry<> registry;
  173. auto group = registry.group<>(entt::get<int, char>);
  174. const auto e0 = registry.create();
  175. registry.assign<int>(e0);
  176. registry.assign<char>(e0);
  177. const auto e1 = registry.create();
  178. registry.assign<int>(e1);
  179. registry.assign<char>(e1);
  180. auto cgroup = std::as_const(registry).group<>(entt::get<const int, const char>);
  181. std::size_t cnt = 0;
  182. group.each([&cnt](auto, int &, char &) { ++cnt; });
  183. group.each([&cnt](int &, char &) { ++cnt; });
  184. ASSERT_EQ(cnt, std::size_t{4});
  185. cgroup.each([&cnt](auto, const int &, const char &) { --cnt; });
  186. cgroup.each([&cnt](const int &, const char &) { --cnt; });
  187. ASSERT_EQ(cnt, std::size_t{0});
  188. }
  189. TEST(OwningGroup, Each) {
  190. entt::registry<> registry;
  191. auto group = registry.group<int>(entt::get<char>);
  192. const auto e0 = registry.create();
  193. registry.assign<int>(e0);
  194. registry.assign<char>(e0);
  195. const auto e1 = registry.create();
  196. registry.assign<int>(e1);
  197. registry.assign<char>(e1);
  198. auto cgroup = std::as_const(registry).group<const int>(entt::get<const char>);
  199. std::size_t cnt = 0;
  200. group.each([&cnt](auto, int &, char &) { ++cnt; });
  201. group.each([&cnt](int &, char &) { ++cnt; });
  202. ASSERT_EQ(cnt, std::size_t{4});
  203. cgroup.each([&cnt](auto, const int &, const char &) { --cnt; });
  204. cgroup.each([&cnt](const int &, const char &) { --cnt; });
  205. ASSERT_EQ(cnt, std::size_t{0});
  206. }
  207. TEST(NonOwningGroup, Sort) {
  208. entt::registry<> registry;
  209. auto group = registry.group<>(entt::get<const int, unsigned int>);
  210. const auto e0 = registry.create();
  211. const auto e1 = registry.create();
  212. const auto e2 = registry.create();
  213. auto uval = 0u;
  214. auto ival = 0;
  215. registry.assign<unsigned int>(e0, uval++);
  216. registry.assign<unsigned int>(e1, uval++);
  217. registry.assign<unsigned int>(e2, uval++);
  218. registry.assign<int>(e0, ival++);
  219. registry.assign<int>(e1, ival++);
  220. registry.assign<int>(e2, ival++);
  221. for(auto entity: group) {
  222. ASSERT_EQ(group.get<unsigned int>(entity), --uval);
  223. ASSERT_EQ(group.get<const int>(entity), --ival);
  224. }
  225. registry.sort<unsigned int>(std::less<unsigned int>{});
  226. group.sort<unsigned int>();
  227. for(auto entity: group) {
  228. ASSERT_EQ(group.get<unsigned int>(entity), uval++);
  229. ASSERT_EQ(group.get<const int>(entity), ival++);
  230. }
  231. }
  232. TEST(NonOwningGroup, IndexRebuiltOnDestroy) {
  233. entt::registry<> registry;
  234. auto group = registry.group<>(entt::get<int, unsigned int>);
  235. const auto e0 = registry.create();
  236. const auto e1 = registry.create();
  237. registry.assign<unsigned int>(e0, 0u);
  238. registry.assign<unsigned int>(e1, 1u);
  239. registry.assign<int>(e0, 0);
  240. registry.assign<int>(e1, 1);
  241. registry.destroy(e0);
  242. registry.assign<int>(registry.create(), 42);
  243. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  244. ASSERT_EQ(group[{}], e1);
  245. ASSERT_EQ(group.get<int>(e1), 1);
  246. ASSERT_EQ(group.get<unsigned int>(e1), 1u);
  247. group.each([e1](auto entity, auto ivalue, auto uivalue) {
  248. ASSERT_EQ(entity, e1);
  249. ASSERT_EQ(ivalue, 1);
  250. ASSERT_EQ(uivalue, 1u);
  251. });
  252. }
  253. TEST(OwningGroup, IndexRebuiltOnDestroy) {
  254. entt::registry<> registry;
  255. auto group = registry.group<int>(entt::get<unsigned int>);
  256. const auto e0 = registry.create();
  257. const auto e1 = registry.create();
  258. registry.assign<unsigned int>(e0, 0u);
  259. registry.assign<unsigned int>(e1, 1u);
  260. registry.assign<int>(e0, 0);
  261. registry.assign<int>(e1, 1);
  262. registry.destroy(e0);
  263. registry.assign<int>(registry.create(), 42);
  264. ASSERT_EQ(group.size(), typename decltype(group)::size_type{1});
  265. ASSERT_EQ(group[{}], e1);
  266. ASSERT_EQ(group.get<int>(e1), 1);
  267. ASSERT_EQ(group.get<unsigned int>(e1), 1u);
  268. group.each([e1](auto entity, auto ivalue, auto uivalue) {
  269. ASSERT_EQ(entity, e1);
  270. ASSERT_EQ(ivalue, 1);
  271. ASSERT_EQ(uivalue, 1u);
  272. });
  273. }
  274. TEST(NonOwningGroup, ConstNonConstAndAllInBetween) {
  275. entt::registry<> registry;
  276. auto group = registry.group<>(entt::get<int, const char>);
  277. ASSERT_TRUE((std::is_same_v<decltype(group.get<int>(0)), int &>));
  278. ASSERT_TRUE((std::is_same_v<decltype(group.get<const int>(0)), const int &>));
  279. ASSERT_TRUE((std::is_same_v<decltype(group.get<const char>(0)), const char &>));
  280. ASSERT_TRUE((std::is_same_v<decltype(group.get<int, const char>(0)), std::tuple<int &, const char &>>));
  281. ASSERT_TRUE((std::is_same_v<decltype(group.get<const int, const char>(0)), std::tuple<const int &, const char &>>));
  282. group.each([](auto, auto &&i, auto &&c) {
  283. ASSERT_TRUE((std::is_same_v<decltype(i), int &>));
  284. ASSERT_TRUE((std::is_same_v<decltype(c), const char &>));
  285. });
  286. }
  287. TEST(OwningGroup, ConstNonConstAndAllInBetween) {
  288. entt::registry<> registry;
  289. auto group = registry.group<int, const char>(entt::get<double, const float>);
  290. ASSERT_TRUE((std::is_same_v<decltype(group.get<int>(0)), int &>));
  291. ASSERT_TRUE((std::is_same_v<decltype(group.get<const int>(0)), const int &>));
  292. ASSERT_TRUE((std::is_same_v<decltype(group.get<const char>(0)), const char &>));
  293. ASSERT_TRUE((std::is_same_v<decltype(group.get<double>(0)), double &>));
  294. ASSERT_TRUE((std::is_same_v<decltype(group.get<const double>(0)), const double &>));
  295. ASSERT_TRUE((std::is_same_v<decltype(group.get<const float>(0)), const float &>));
  296. ASSERT_TRUE((std::is_same_v<decltype(group.get<int, const char, double, const float>(0)), std::tuple<int &, const char &, double &, const float &>>));
  297. ASSERT_TRUE((std::is_same_v<decltype(group.get<const int, const char, const double, const float>(0)), std::tuple<const int &, const char &, const double &, const float &>>));
  298. group.each([](auto, auto &&i, auto &&c, auto &&d, auto &&f) {
  299. ASSERT_TRUE((std::is_same_v<decltype(i), int &>));
  300. ASSERT_TRUE((std::is_same_v<decltype(c), const char &>));
  301. ASSERT_TRUE((std::is_same_v<decltype(d), double &>));
  302. ASSERT_TRUE((std::is_same_v<decltype(f), const float &>));
  303. });
  304. }
  305. TEST(NonOwningGroup, Find) {
  306. entt::registry<> registry;
  307. auto group = registry.group<>(entt::get<int, const char>);
  308. const auto e0 = registry.create();
  309. registry.assign<int>(e0);
  310. registry.assign<char>(e0);
  311. const auto e1 = registry.create();
  312. registry.assign<int>(e1);
  313. registry.assign<char>(e1);
  314. const auto e2 = registry.create();
  315. registry.assign<int>(e2);
  316. registry.assign<char>(e2);
  317. const auto e3 = registry.create();
  318. registry.assign<int>(e3);
  319. registry.assign<char>(e3);
  320. registry.remove<int>(e1);
  321. ASSERT_NE(group.find(e0), group.end());
  322. ASSERT_EQ(group.find(e1), group.end());
  323. ASSERT_NE(group.find(e2), group.end());
  324. ASSERT_NE(group.find(e3), group.end());
  325. auto it = group.find(e2);
  326. ASSERT_EQ(*it, e2);
  327. ASSERT_EQ(*(++it), e3);
  328. ASSERT_EQ(*(++it), e0);
  329. ASSERT_EQ(++it, group.end());
  330. ASSERT_EQ(++group.find(e0), group.end());
  331. const auto e4 = registry.create();
  332. registry.destroy(e4);
  333. const auto e5 = registry.create();
  334. registry.assign<int>(e5);
  335. registry.assign<char>(e5);
  336. ASSERT_NE(group.find(e5), group.end());
  337. ASSERT_EQ(group.find(e4), group.end());
  338. }
  339. TEST(OwningGroup, Find) {
  340. entt::registry<> registry;
  341. auto group = registry.group<int>(entt::get<const char>);
  342. const auto e0 = registry.create();
  343. registry.assign<int>(e0);
  344. registry.assign<char>(e0);
  345. const auto e1 = registry.create();
  346. registry.assign<int>(e1);
  347. registry.assign<char>(e1);
  348. const auto e2 = registry.create();
  349. registry.assign<int>(e2);
  350. registry.assign<char>(e2);
  351. const auto e3 = registry.create();
  352. registry.assign<int>(e3);
  353. registry.assign<char>(e3);
  354. registry.remove<int>(e1);
  355. ASSERT_NE(group.find(e0), group.end());
  356. ASSERT_EQ(group.find(e1), group.end());
  357. ASSERT_NE(group.find(e2), group.end());
  358. ASSERT_NE(group.find(e3), group.end());
  359. auto it = group.find(e2);
  360. ASSERT_EQ(*it, e2);
  361. ASSERT_EQ(*(++it), e3);
  362. ASSERT_EQ(*(++it), e0);
  363. ASSERT_EQ(++it, group.end());
  364. ASSERT_EQ(++group.find(e0), group.end());
  365. const auto e4 = registry.create();
  366. registry.destroy(e4);
  367. const auto e5 = registry.create();
  368. registry.assign<int>(e5);
  369. registry.assign<char>(e5);
  370. ASSERT_NE(group.find(e5), group.end());
  371. ASSERT_EQ(group.find(e4), group.end());
  372. }
  373. TEST(NonOwningGroup, ExcludedComponents) {
  374. entt::registry<> registry;
  375. const auto e0 = registry.create();
  376. registry.assign<int>(e0, 0);
  377. const auto e1 = registry.create();
  378. registry.assign<int>(e1, 1);
  379. registry.assign<char>(e1);
  380. const auto group = registry.group<>(entt::get<int>, entt::exclude<char>);
  381. const auto e2 = registry.create();
  382. registry.assign<int>(e2, 2);
  383. const auto e3 = registry.create();
  384. registry.assign<int>(e3, 3);
  385. registry.assign<char>(e3);
  386. for(const auto entity: group) {
  387. if(entity == e0) {
  388. ASSERT_EQ(group.get<int>(e0), 0);
  389. } else if(entity == e2) {
  390. ASSERT_EQ(group.get<int>(e2), 2);
  391. } else {
  392. FAIL();
  393. }
  394. }
  395. registry.assign<char>(e0);
  396. registry.assign<char>(e2);
  397. ASSERT_TRUE(group.empty());
  398. registry.remove<char>(e1);
  399. registry.remove<char>(e3);
  400. for(const auto entity: group) {
  401. if(entity == e1) {
  402. ASSERT_EQ(group.get<int>(e1), 1);
  403. } else if(entity == e3) {
  404. ASSERT_EQ(group.get<int>(e3), 3);
  405. } else {
  406. FAIL();
  407. }
  408. }
  409. }
  410. TEST(OwningGroup, ExcludedComponents) {
  411. entt::registry<> registry;
  412. const auto e0 = registry.create();
  413. registry.assign<int>(e0, 0);
  414. const auto e1 = registry.create();
  415. registry.assign<int>(e1, 1);
  416. registry.assign<char>(e1);
  417. const auto group = registry.group<int>(entt::exclude<char>);
  418. const auto e2 = registry.create();
  419. registry.assign<int>(e2, 2);
  420. const auto e3 = registry.create();
  421. registry.assign<int>(e3, 3);
  422. registry.assign<char>(e3);
  423. for(const auto entity: group) {
  424. if(entity == e0) {
  425. ASSERT_EQ(group.get<int>(e0), 0);
  426. } else if(entity == e2) {
  427. ASSERT_EQ(group.get<int>(e2), 2);
  428. } else {
  429. FAIL();
  430. }
  431. }
  432. registry.assign<char>(e0);
  433. registry.assign<char>(e2);
  434. ASSERT_TRUE(group.empty());
  435. registry.remove<char>(e1);
  436. registry.remove<char>(e3);
  437. for(const auto entity: group) {
  438. if(entity == e1) {
  439. ASSERT_EQ(group.get<int>(e1), 1);
  440. } else if(entity == e3) {
  441. ASSERT_EQ(group.get<int>(e3), 3);
  442. } else {
  443. FAIL();
  444. }
  445. }
  446. }
  447. TEST(NonOwningGroup, EmptyAndNonEmptyTypes) {
  448. struct empty_type {};
  449. entt::registry<> registry;
  450. const auto group = registry.group<>(entt::get<int, empty_type>);
  451. const auto e0 = registry.create();
  452. registry.assign<empty_type>(e0);
  453. registry.assign<int>(e0);
  454. const auto e1 = registry.create();
  455. registry.assign<empty_type>(e1);
  456. registry.assign<int>(e1);
  457. registry.assign<int>(registry.create());
  458. for(const auto entity: group) {
  459. ASSERT_TRUE(entity == e0 || entity == e1);
  460. }
  461. group.each([e0, e1](const auto entity, const int &, const empty_type &) {
  462. ASSERT_TRUE(entity == e0 || entity == e1);
  463. });
  464. ASSERT_EQ(group.size(), typename decltype(group)::size_type{2});
  465. ASSERT_EQ(&group.get<empty_type>(e0), &group.get<empty_type>(e1));
  466. }
  467. TEST(OwningGroup, EmptyAndNonEmptyTypes) {
  468. struct empty_type {};
  469. entt::registry<> registry;
  470. const auto group = registry.group<empty_type>(entt::get<int>);
  471. const auto e0 = registry.create();
  472. registry.assign<empty_type>(e0);
  473. registry.assign<int>(e0);
  474. const auto e1 = registry.create();
  475. registry.assign<empty_type>(e1);
  476. registry.assign<int>(e1);
  477. registry.assign<int>(registry.create());
  478. for(const auto entity: group) {
  479. ASSERT_TRUE(entity == e0 || entity == e1);
  480. }
  481. group.each([e0, e1](const auto entity, const empty_type &, const int &) {
  482. ASSERT_TRUE(entity == e0 || entity == e1);
  483. });
  484. ASSERT_EQ(group.size(), typename decltype(group)::size_type{2});
  485. ASSERT_EQ(&group.get<empty_type>(e0), &group.get<empty_type>(e1));
  486. }
  487. TEST(NonOwningGroup, TrackEntitiesOnComponentDestruction) {
  488. entt::registry<> registry;
  489. const entt::registry<> &cregistry = registry;
  490. const auto group = registry.group<>(entt::get<int>, entt::exclude<char>);
  491. const auto cgroup = cregistry.group<>(entt::get<const int>, entt::exclude<char>);
  492. const auto entity = registry.create();
  493. registry.assign<int>(entity);
  494. registry.assign<char>(entity);
  495. ASSERT_TRUE(group.empty());
  496. ASSERT_TRUE(cgroup.empty());
  497. registry.remove<char>(entity);
  498. ASSERT_FALSE(group.empty());
  499. ASSERT_FALSE(cgroup.empty());
  500. }
  501. TEST(OwningGroup, TrackEntitiesOnComponentDestruction) {
  502. entt::registry<> registry;
  503. const entt::registry<> &cregistry = registry;
  504. const auto group = registry.group<int>(entt::exclude<char>);
  505. const auto cgroup = cregistry.group<const int>(entt::exclude<char>);
  506. const auto entity = registry.create();
  507. registry.assign<int>(entity);
  508. registry.assign<char>(entity);
  509. ASSERT_TRUE(group.empty());
  510. ASSERT_TRUE(cgroup.empty());
  511. registry.remove<char>(entity);
  512. ASSERT_FALSE(group.empty());
  513. ASSERT_FALSE(cgroup.empty());
  514. }