view.cpp 17 KB

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