view.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. #include <utility>
  2. #include <type_traits>
  3. #include <gtest/gtest.h>
  4. #include <entt/entity/helper.hpp>
  5. #include <entt/entity/registry.hpp>
  6. #include <entt/entity/view.hpp>
  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.assign<int>(e1);
  15. registry.assign<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.assign<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.assign<int>(e0);
  44. const auto e1 = registry.create();
  45. registry.assign<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.assign<int>(e0);
  55. const auto e1 = registry.create();
  56. registry.assign<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.assign<char>(e0);
  66. registry.assign<double>(e0);
  67. const auto e1 = registry.create();
  68. registry.assign<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.assign<int>(registry.create());
  76. registry.assign<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.assign<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.assign<int>(e0);
  113. const auto e1 = registry.create();
  114. registry.assign<int>(e1);
  115. const auto e2 = registry.create();
  116. registry.assign<int>(e2);
  117. const auto e3 = registry.create();
  118. registry.assign<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.assign<int>(e5);
  134. ASSERT_NE(view.find(e5), view.end());
  135. ASSERT_EQ(view.find(e4), view.end());
  136. }
  137. TEST(SingleComponentView, Less) {
  138. entt::registry registry;
  139. const auto entity = std::get<0>(registry.create<int, entt::tag<"empty"_hs>>());
  140. registry.create<char>();
  141. registry.view<entt::tag<"empty"_hs>>().less([entity](const auto entt) {
  142. ASSERT_EQ(entity, entt);
  143. });
  144. registry.view<entt::tag<"empty"_hs>>().less([check = true]() mutable {
  145. ASSERT_TRUE(check);
  146. check = false;
  147. });
  148. registry.view<int>().less([entity](const auto entt, int) {
  149. ASSERT_EQ(entity, entt);
  150. });
  151. registry.view<int>().less([check = true](int) mutable {
  152. ASSERT_TRUE(check);
  153. check = false;
  154. });
  155. }
  156. TEST(MultiComponentView, Functionalities) {
  157. entt::registry registry;
  158. auto view = registry.view<int, char>();
  159. auto cview = std::as_const(registry).view<const int, const char>();
  160. ASSERT_TRUE(view.empty());
  161. ASSERT_TRUE((view.empty<int, char>()));
  162. ASSERT_TRUE((cview.empty<const int, const char>()));
  163. const auto e0 = registry.create();
  164. registry.assign<char>(e0);
  165. const auto e1 = registry.create();
  166. registry.assign<int>(e1);
  167. ASSERT_FALSE(view.empty());
  168. ASSERT_FALSE((view.empty<int>()));
  169. ASSERT_FALSE((cview.empty<const char>()));
  170. registry.assign<char>(e1);
  171. auto it = registry.view<int, char>().begin();
  172. ASSERT_EQ(*it, e1);
  173. ASSERT_EQ(++it, (registry.view<int, char>().end()));
  174. ASSERT_NO_THROW((registry.view<int, char>().begin()++));
  175. ASSERT_NO_THROW((++registry.view<int, char>().begin()));
  176. ASSERT_NE(view.begin(), view.end());
  177. ASSERT_NE(cview.begin(), cview.end());
  178. ASSERT_EQ(view.size(), decltype(view.size()){1});
  179. ASSERT_EQ(view.size<int>(), decltype(view.size()){1});
  180. ASSERT_EQ(cview.size<const char>(), decltype(view.size()){2});
  181. registry.get<char>(e0) = '1';
  182. registry.get<char>(e1) = '2';
  183. registry.get<int>(e1) = 42;
  184. for(auto entity: view) {
  185. ASSERT_EQ(std::get<0>(cview.get<const int, const char>(entity)), 42);
  186. ASSERT_EQ(std::get<1>(view.get<int, char>(entity)), '2');
  187. ASSERT_EQ(cview.get<const char>(entity), '2');
  188. }
  189. ASSERT_EQ(*(view.data<int>() + 0), e1);
  190. ASSERT_EQ(*(view.data<char>() + 0), e0);
  191. ASSERT_EQ(*(cview.data<const char>() + 1), e1);
  192. ASSERT_EQ(*(view.raw<int>() + 0), 42);
  193. ASSERT_EQ(*(view.raw<char>() + 0), '1');
  194. ASSERT_EQ(*(cview.raw<const char>() + 1), '2');
  195. }
  196. TEST(MultiComponentView, Iterator) {
  197. entt::registry registry;
  198. const auto entity = registry.create();
  199. registry.assign<int>(entity);
  200. registry.assign<char>(entity);
  201. const auto view = registry.view<int, char>();
  202. using iterator_type = typename decltype(view)::iterator_type;
  203. iterator_type end{view.begin()};
  204. iterator_type begin{};
  205. begin = view.end();
  206. std::swap(begin, end);
  207. ASSERT_EQ(begin, view.begin());
  208. ASSERT_EQ(end, view.end());
  209. ASSERT_NE(begin, end);
  210. ASSERT_EQ(begin++, view.begin());
  211. ASSERT_EQ(begin--, view.end());
  212. ASSERT_EQ(++begin, view.end());
  213. ASSERT_EQ(--begin, view.begin());
  214. ASSERT_EQ(*begin, entity);
  215. ASSERT_EQ(*begin.operator->(), entity);
  216. }
  217. TEST(MultiComponentView, Contains) {
  218. entt::registry registry;
  219. const auto e0 = registry.create();
  220. registry.assign<int>(e0);
  221. registry.assign<char>(e0);
  222. const auto e1 = registry.create();
  223. registry.assign<int>(e1);
  224. registry.assign<char>(e1);
  225. registry.destroy(e0);
  226. auto view = registry.view<int, char>();
  227. ASSERT_FALSE(view.contains(e0));
  228. ASSERT_TRUE(view.contains(e1));
  229. }
  230. TEST(MultiComponentView, Empty) {
  231. entt::registry registry;
  232. const auto e0 = registry.create();
  233. registry.assign<double>(e0);
  234. registry.assign<int>(e0);
  235. registry.assign<float>(e0);
  236. const auto e1 = registry.create();
  237. registry.assign<char>(e1);
  238. registry.assign<float>(e1);
  239. auto view = registry.view<char, int, float>();
  240. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  241. ASSERT_EQ(view.begin(), view.end());
  242. }
  243. TEST(MultiComponentView, Each) {
  244. entt::registry registry;
  245. const auto e0 = registry.create();
  246. registry.assign<int>(e0);
  247. registry.assign<char>(e0);
  248. const auto e1 = registry.create();
  249. registry.assign<int>(e1);
  250. registry.assign<char>(e1);
  251. auto view = registry.view<int, char>();
  252. auto cview = std::as_const(registry).view<const int, const char>();
  253. std::size_t cnt = 0;
  254. view.each([&cnt](auto, int &, char &) { ++cnt; });
  255. view.each([&cnt](int &, char &) { ++cnt; });
  256. ASSERT_EQ(cnt, std::size_t{4});
  257. cview.each([&cnt](auto, const int &, const char &) { --cnt; });
  258. cview.each([&cnt](const int &, const char &) { --cnt; });
  259. ASSERT_EQ(cnt, std::size_t{0});
  260. }
  261. TEST(MultiComponentView, EachWithType) {
  262. entt::registry registry;
  263. for(auto i = 0; i < 3; ++i) {
  264. const auto entity = registry.create();
  265. registry.assign<int>(entity, i);
  266. registry.assign<char>(entity);
  267. }
  268. // makes char a better candidate during iterations
  269. const auto entity = registry.create();
  270. registry.assign<int>(entity, 99);
  271. registry.view<int, char>().each<int>([value = 2](const auto curr, const auto) mutable {
  272. ASSERT_EQ(curr, value--);
  273. });
  274. registry.sort<int>([](const auto lhs, const auto rhs) {
  275. return lhs < rhs;
  276. });
  277. registry.view<int, char>().each<int>([value = 0](const auto curr, const auto) mutable {
  278. ASSERT_EQ(curr, value++);
  279. });
  280. }
  281. TEST(MultiComponentView, EachWithHoles) {
  282. entt::registry registry;
  283. const auto e0 = registry.create();
  284. const auto e1 = registry.create();
  285. const auto e2 = registry.create();
  286. registry.assign<char>(e0, '0');
  287. registry.assign<char>(e1, '1');
  288. registry.assign<int>(e0, 0);
  289. registry.assign<int>(e2, 2);
  290. auto view = registry.view<char, int>();
  291. view.each([e0](auto entity, const char &c, const int &i) {
  292. ASSERT_EQ(entity, e0);
  293. ASSERT_EQ(c, '0');
  294. ASSERT_EQ(i, 0);
  295. });
  296. }
  297. TEST(MultiComponentView, ConstNonConstAndAllInBetween) {
  298. entt::registry registry;
  299. auto view = registry.view<int, const char, std::true_type>();
  300. ASSERT_EQ(view.size(), decltype(view.size()){0});
  301. const auto entity = registry.create();
  302. registry.assign<int>(entity, 0);
  303. registry.assign<char>(entity, 'c');
  304. registry.assign<std::true_type>(entity);
  305. ASSERT_EQ(view.size(), decltype(view.size()){1});
  306. ASSERT_TRUE((std::is_same_v<decltype(view.get<int>({})), int &>));
  307. ASSERT_TRUE((std::is_same_v<decltype(view.get<const char>({})), const char &>));
  308. ASSERT_TRUE((std::is_same_v<decltype(view.get<std::true_type>({})), std::true_type>));
  309. ASSERT_TRUE((std::is_same_v<decltype(view.get<int, const char, std::true_type>({})), std::tuple<int &, const char &, std::true_type>>));
  310. ASSERT_TRUE((std::is_same_v<decltype(view.raw<const char>()), const char *>));
  311. ASSERT_TRUE((std::is_same_v<decltype(view.raw<int>()), int *>));
  312. view.each([](auto &&i, auto &&c, auto &&e) {
  313. ASSERT_TRUE((std::is_same_v<decltype(i), int &>));
  314. ASSERT_TRUE((std::is_same_v<decltype(c), const char &>));
  315. ASSERT_TRUE((std::is_same_v<decltype(e), std::true_type &&>));
  316. });
  317. }
  318. TEST(MultiComponentView, Find) {
  319. entt::registry registry;
  320. auto view = registry.view<int, const char>();
  321. const auto e0 = registry.create();
  322. registry.assign<int>(e0);
  323. registry.assign<char>(e0);
  324. const auto e1 = registry.create();
  325. registry.assign<int>(e1);
  326. registry.assign<char>(e1);
  327. const auto e2 = registry.create();
  328. registry.assign<int>(e2);
  329. registry.assign<char>(e2);
  330. const auto e3 = registry.create();
  331. registry.assign<int>(e3);
  332. registry.assign<char>(e3);
  333. registry.remove<int>(e1);
  334. ASSERT_NE(view.find(e0), view.end());
  335. ASSERT_EQ(view.find(e1), view.end());
  336. ASSERT_NE(view.find(e2), view.end());
  337. ASSERT_NE(view.find(e3), view.end());
  338. auto it = view.find(e2);
  339. ASSERT_EQ(*it, e2);
  340. ASSERT_EQ(*(++it), e3);
  341. ASSERT_EQ(*(++it), e0);
  342. ASSERT_EQ(++it, view.end());
  343. ASSERT_EQ(++view.find(e0), view.end());
  344. const auto e4 = registry.create();
  345. registry.destroy(e4);
  346. const auto e5 = registry.create();
  347. registry.assign<int>(e5);
  348. registry.assign<char>(e5);
  349. ASSERT_NE(view.find(e5), view.end());
  350. ASSERT_EQ(view.find(e4), view.end());
  351. }
  352. TEST(MultiComponentView, ExcludedComponents) {
  353. entt::registry registry;
  354. const auto e0 = registry.create();
  355. registry.assign<int>(e0, 0);
  356. const auto e1 = registry.create();
  357. registry.assign<int>(e1, 1);
  358. registry.assign<char>(e1);
  359. const auto view = registry.view<int>(entt::exclude<char>);
  360. const auto e2 = registry.create();
  361. registry.assign<int>(e2, 2);
  362. const auto e3 = registry.create();
  363. registry.assign<int>(e3, 3);
  364. registry.assign<char>(e3);
  365. for(const auto entity: view) {
  366. ASSERT_TRUE(entity == e0 || entity == e2);
  367. if(entity == e0) {
  368. ASSERT_EQ(view.get<int>(e0), 0);
  369. } else if(entity == e2) {
  370. ASSERT_EQ(view.get(e2), 2);
  371. }
  372. }
  373. registry.assign<char>(e0);
  374. registry.assign<char>(e2);
  375. registry.remove<char>(e1);
  376. registry.remove<char>(e3);
  377. for(const auto entity: view) {
  378. ASSERT_TRUE(entity == e1 || entity == e3);
  379. if(entity == e1) {
  380. ASSERT_EQ(view.get(e1), 1);
  381. } else if(entity == e3) {
  382. ASSERT_EQ(view.get<int>(e3), 3);
  383. }
  384. }
  385. }
  386. TEST(MultiComponentView, Less) {
  387. entt::registry registry;
  388. const auto entity = std::get<0>(registry.create<int, char, double, entt::tag<"empty"_hs>>());
  389. registry.create<int, char>();
  390. registry.view<int, char, entt::tag<"empty"_hs>>().less([entity](const auto entt, int, char) {
  391. ASSERT_EQ(entity, entt);
  392. });
  393. registry.view<int, entt::tag<"empty"_hs>, char>().less([check = true](int, char) mutable {
  394. ASSERT_TRUE(check);
  395. check = false;
  396. });
  397. registry.view<entt::tag<"empty"_hs>, int, char>().less([entity](const auto entt, int, char) {
  398. ASSERT_EQ(entity, entt);
  399. });
  400. registry.view<entt::tag<"empty"_hs>, int, char>().less<entt::tag<"empty"_hs>>([entity](const auto entt, int, char) {
  401. ASSERT_EQ(entity, entt);
  402. });
  403. registry.view<int, entt::tag<"empty"_hs>, char>().less<entt::tag<"empty"_hs>>([check = true](int, char) mutable {
  404. ASSERT_TRUE(check);
  405. check = false;
  406. });
  407. registry.view<int, char, double>().less([entity](const auto entt, int, char, double) {
  408. ASSERT_EQ(entity, entt);
  409. });
  410. }