view.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  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(e0) = '1';
  25. view.get(e1) = '2';
  26. for(auto entity: view) {
  27. ASSERT_TRUE(cview.get(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(entt::entity{0})), int &>));
  98. ASSERT_TRUE((std::is_same_v<decltype(view.raw()), int *>));
  99. ASSERT_TRUE((std::is_same_v<decltype(cview.get(entt::entity{0})), const int &>));
  100. ASSERT_TRUE((std::is_same_v<decltype(cview.raw()), const int *>));
  101. view.each([](auto, auto &&i) {
  102. ASSERT_TRUE((std::is_same_v<decltype(i), int &>));
  103. });
  104. cview.each([](auto, 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(MultipleComponentView, 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>());
  162. ASSERT_TRUE(cview.empty<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(MultipleComponentView, 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(view.begin()++, view.begin());
  211. ASSERT_EQ(++view.begin(), view.end());
  212. }
  213. TEST(MultipleComponentView, Contains) {
  214. entt::registry registry;
  215. const auto e0 = registry.create();
  216. registry.assign<int>(e0);
  217. registry.assign<char>(e0);
  218. const auto e1 = registry.create();
  219. registry.assign<int>(e1);
  220. registry.assign<char>(e1);
  221. registry.destroy(e0);
  222. auto view = registry.view<int, char>();
  223. ASSERT_FALSE(view.contains(e0));
  224. ASSERT_TRUE(view.contains(e1));
  225. }
  226. TEST(MultipleComponentView, Empty) {
  227. entt::registry registry;
  228. const auto e0 = registry.create();
  229. registry.assign<double>(e0);
  230. registry.assign<int>(e0);
  231. registry.assign<float>(e0);
  232. const auto e1 = registry.create();
  233. registry.assign<char>(e1);
  234. registry.assign<float>(e1);
  235. auto view = registry.view<char, int, float>();
  236. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  237. ASSERT_EQ(view.begin(), view.end());
  238. }
  239. TEST(MultipleComponentView, Each) {
  240. entt::registry registry;
  241. const auto e0 = registry.create();
  242. registry.assign<int>(e0);
  243. registry.assign<char>(e0);
  244. const auto e1 = registry.create();
  245. registry.assign<int>(e1);
  246. registry.assign<char>(e1);
  247. auto view = registry.view<int, char>();
  248. auto cview = std::as_const(registry).view<const int, const char>();
  249. std::size_t cnt = 0;
  250. view.each([&cnt](auto, int &, char &) { ++cnt; });
  251. view.each([&cnt](int &, char &) { ++cnt; });
  252. ASSERT_EQ(cnt, std::size_t{4});
  253. cview.each([&cnt](auto, const int &, const char &) { --cnt; });
  254. cview.each([&cnt](const int &, const char &) { --cnt; });
  255. ASSERT_EQ(cnt, std::size_t{0});
  256. }
  257. TEST(MultipleComponentView, EachWithType) {
  258. entt::registry registry;
  259. for(auto i = 0; i < 3; ++i) {
  260. const auto entity = registry.create();
  261. registry.assign<int>(entity, i);
  262. registry.assign<char>(entity);
  263. }
  264. // makes char a better candidate during iterations
  265. const auto entity = registry.create();
  266. registry.assign<int>(entity, 99);
  267. registry.view<int, char>().each<int>([value = 2](const auto curr, const auto) mutable {
  268. ASSERT_EQ(curr, value--);
  269. });
  270. registry.sort<int>([](const auto lhs, const auto rhs) {
  271. return lhs < rhs;
  272. });
  273. registry.view<int, char>().each<int>([value = 0](const auto curr, const auto) mutable {
  274. ASSERT_EQ(curr, value++);
  275. });
  276. }
  277. TEST(MultipleComponentView, EachWithHoles) {
  278. entt::registry registry;
  279. const auto e0 = registry.create();
  280. const auto e1 = registry.create();
  281. const auto e2 = registry.create();
  282. registry.assign<char>(e0, '0');
  283. registry.assign<char>(e1, '1');
  284. registry.assign<int>(e0, 0);
  285. registry.assign<int>(e2, 2);
  286. auto view = registry.view<char, int>();
  287. view.each([e0](auto entity, const char &c, const int &i) {
  288. if(e0 == entity) {
  289. ASSERT_EQ(c, '0');
  290. ASSERT_EQ(i, 0);
  291. } else {
  292. FAIL();
  293. }
  294. });
  295. }
  296. TEST(MultipleComponentView, ConstNonConstAndAllInBetween) {
  297. entt::registry registry;
  298. auto view = registry.view<int, const char>();
  299. ASSERT_EQ(view.size(), decltype(view.size()){0});
  300. const auto entity = registry.create();
  301. registry.assign<int>(entity, 0);
  302. registry.assign<char>(entity, 'c');
  303. ASSERT_EQ(view.size(), decltype(view.size()){1});
  304. ASSERT_TRUE((std::is_same_v<decltype(view.get<int>(entt::entity{0})), int &>));
  305. ASSERT_TRUE((std::is_same_v<decltype(view.get<const char>(entt::entity{0})), const char &>));
  306. ASSERT_TRUE((std::is_same_v<decltype(view.get<int, const char>(entt::entity{0})), std::tuple<int &, const char &>>));
  307. ASSERT_TRUE((std::is_same_v<decltype(view.raw<const char>()), const char *>));
  308. ASSERT_TRUE((std::is_same_v<decltype(view.raw<int>()), int *>));
  309. view.each([](auto, auto &&i, auto &&c) {
  310. ASSERT_TRUE((std::is_same_v<decltype(i), int &>));
  311. ASSERT_TRUE((std::is_same_v<decltype(c), const char &>));
  312. });
  313. }
  314. TEST(MultipleComponentView, Find) {
  315. entt::registry registry;
  316. auto view = registry.view<int, const char>();
  317. const auto e0 = registry.create();
  318. registry.assign<int>(e0);
  319. registry.assign<char>(e0);
  320. const auto e1 = registry.create();
  321. registry.assign<int>(e1);
  322. registry.assign<char>(e1);
  323. const auto e2 = registry.create();
  324. registry.assign<int>(e2);
  325. registry.assign<char>(e2);
  326. const auto e3 = registry.create();
  327. registry.assign<int>(e3);
  328. registry.assign<char>(e3);
  329. registry.remove<int>(e1);
  330. ASSERT_NE(view.find(e0), view.end());
  331. ASSERT_EQ(view.find(e1), view.end());
  332. ASSERT_NE(view.find(e2), view.end());
  333. ASSERT_NE(view.find(e3), view.end());
  334. auto it = view.find(e2);
  335. ASSERT_EQ(*it, e2);
  336. ASSERT_EQ(*(++it), e3);
  337. ASSERT_EQ(*(++it), e0);
  338. ASSERT_EQ(++it, view.end());
  339. ASSERT_EQ(++view.find(e0), view.end());
  340. const auto e4 = registry.create();
  341. registry.destroy(e4);
  342. const auto e5 = registry.create();
  343. registry.assign<int>(e5);
  344. registry.assign<char>(e5);
  345. ASSERT_NE(view.find(e5), view.end());
  346. ASSERT_EQ(view.find(e4), view.end());
  347. }
  348. TEST(MultiComponentView, Less) {
  349. entt::registry registry;
  350. const auto entity = std::get<0>(registry.create<int, char, double, entt::tag<"empty"_hs>>());
  351. registry.create<int, char>();
  352. registry.view<int, char, entt::tag<"empty"_hs>>().less([entity](const auto entt, int, char) {
  353. ASSERT_EQ(entity, entt);
  354. });
  355. registry.view<int, entt::tag<"empty"_hs>, char>().less([check = true](int, char) mutable {
  356. ASSERT_TRUE(check);
  357. check = false;
  358. });
  359. registry.view<entt::tag<"empty"_hs>, int, char>().less([entity](const auto entt, int, char) {
  360. ASSERT_EQ(entity, entt);
  361. });
  362. registry.view<entt::tag<"empty"_hs>, int, char>().less<entt::tag<"empty"_hs>>([entity](const auto entt, int, char) {
  363. ASSERT_EQ(entity, entt);
  364. });
  365. registry.view<int, entt::tag<"empty"_hs>, char>().less<entt::tag<"empty"_hs>>([check = true](int, char) mutable {
  366. ASSERT_TRUE(check);
  367. check = false;
  368. });
  369. registry.view<int, char, double>().less([entity](const auto entt, int, char, double) {
  370. ASSERT_EQ(entity, entt);
  371. });
  372. }