view.cpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842
  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, Proxy) {
  74. entt::registry registry;
  75. registry.emplace<int>(registry.create());
  76. registry.emplace<int>(registry.create());
  77. auto view = registry.view<int>();
  78. auto cview = std::as_const(registry).view<const int>();
  79. std::size_t cnt = 0;
  80. view.each([&cnt](auto, int &) { ++cnt; });
  81. view.each([&cnt](int &) { ++cnt; });
  82. for(auto &&[entt, iv]: view.proxy()) {
  83. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  84. static_assert(std::is_same_v<decltype(iv), int &>);
  85. ++cnt;
  86. }
  87. ASSERT_EQ(cnt, std::size_t{6});
  88. cview.each([&cnt](auto, const int &) { --cnt; });
  89. cview.each([&cnt](const int &) { --cnt; });
  90. for(auto &&[entt, iv]: cview.proxy()) {
  91. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  92. static_assert(std::is_same_v<decltype(iv), const int &>);
  93. --cnt;
  94. }
  95. ASSERT_EQ(cnt, std::size_t{0});
  96. }
  97. TEST(SingleComponentView, ConstNonConstAndAllInBetween) {
  98. entt::registry registry;
  99. auto view = registry.view<int>();
  100. auto cview = std::as_const(registry).view<const int>();
  101. ASSERT_EQ(view.size(), decltype(view.size()){0});
  102. ASSERT_EQ(cview.size(), decltype(cview.size()){0});
  103. registry.emplace<int>(registry.create(), 0);
  104. ASSERT_EQ(view.size(), decltype(view.size()){1});
  105. ASSERT_EQ(cview.size(), decltype(cview.size()){1});
  106. static_assert(std::is_same_v<typename decltype(view)::raw_type, int>);
  107. static_assert(std::is_same_v<typename decltype(cview)::raw_type, const int>);
  108. static_assert(std::is_same_v<decltype(view.get({})), int &>);
  109. static_assert(std::is_same_v<decltype(view.raw()), int *>);
  110. static_assert(std::is_same_v<decltype(cview.get({})), const int &>);
  111. static_assert(std::is_same_v<decltype(cview.raw()), const int *>);
  112. view.each([](auto &&i) {
  113. static_assert(std::is_same_v<decltype(i), int &>);
  114. });
  115. cview.each([](auto &&i) {
  116. static_assert(std::is_same_v<decltype(i), const int &>);
  117. });
  118. for(auto &&[entt, iv]: view.proxy()) {
  119. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  120. static_assert(std::is_same_v<decltype(iv), int &>);
  121. }
  122. for(auto &&[entt, iv]: cview.proxy()) {
  123. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  124. static_assert(std::is_same_v<decltype(iv), const int &>);
  125. }
  126. }
  127. TEST(SingleComponentView, Find) {
  128. entt::registry registry;
  129. auto view = registry.view<int>();
  130. const auto e0 = registry.create();
  131. registry.emplace<int>(e0);
  132. const auto e1 = registry.create();
  133. registry.emplace<int>(e1);
  134. const auto e2 = registry.create();
  135. registry.emplace<int>(e2);
  136. const auto e3 = registry.create();
  137. registry.emplace<int>(e3);
  138. registry.remove<int>(e1);
  139. ASSERT_NE(view.find(e0), view.end());
  140. ASSERT_EQ(view.find(e1), view.end());
  141. ASSERT_NE(view.find(e2), view.end());
  142. ASSERT_NE(view.find(e3), view.end());
  143. auto it = view.find(e2);
  144. ASSERT_EQ(*it, e2);
  145. ASSERT_EQ(*(++it), e3);
  146. ASSERT_EQ(*(++it), e0);
  147. ASSERT_EQ(++it, view.end());
  148. ASSERT_EQ(++view.find(e0), view.end());
  149. const auto e4 = registry.create();
  150. registry.destroy(e4);
  151. const auto e5 = registry.create();
  152. registry.emplace<int>(e5);
  153. ASSERT_NE(view.find(e5), view.end());
  154. ASSERT_EQ(view.find(e4), view.end());
  155. }
  156. TEST(SingleComponentView, EmptyTypes) {
  157. entt::registry registry;
  158. auto create = [&](auto... component) {
  159. const auto entt = registry.create();
  160. (registry.emplace<decltype(component)>(entt, component), ...);
  161. return entt;
  162. };
  163. const auto entity = create(0, empty_type{});
  164. create('c');
  165. registry.view<empty_type>().each([entity](const auto entt) {
  166. ASSERT_EQ(entity, entt);
  167. });
  168. registry.view<empty_type>().each([check = true]() mutable {
  169. ASSERT_TRUE(check);
  170. check = false;
  171. });
  172. for(auto &&[entt]: registry.view<empty_type>().proxy()) {
  173. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  174. ASSERT_EQ(entity, entt);
  175. }
  176. registry.view<int>().each([entity](const auto entt, int) {
  177. ASSERT_EQ(entity, entt);
  178. });
  179. registry.view<int>().each([check = true](int) mutable {
  180. ASSERT_TRUE(check);
  181. check = false;
  182. });
  183. for(auto &&[entt, iv]: registry.view<int>().proxy()) {
  184. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  185. static_assert(std::is_same_v<decltype(iv), int &>);
  186. ASSERT_EQ(entity, entt);
  187. }
  188. }
  189. TEST(SingleComponentView, FrontBack) {
  190. entt::registry registry;
  191. auto view = registry.view<const int>();
  192. ASSERT_EQ(view.front(), static_cast<entt::entity>(entt::null));
  193. ASSERT_EQ(view.back(), static_cast<entt::entity>(entt::null));
  194. const auto e0 = registry.create();
  195. registry.emplace<int>(e0);
  196. const auto e1 = registry.create();
  197. registry.emplace<int>(e1);
  198. ASSERT_EQ(view.front(), e1);
  199. ASSERT_EQ(view.back(), e0);
  200. }
  201. TEST(MultiComponentView, Functionalities) {
  202. entt::registry registry;
  203. auto view = registry.view<int, char>();
  204. auto cview = std::as_const(registry).view<const int, const char>();
  205. ASSERT_TRUE(view.empty());
  206. ASSERT_TRUE((view.empty<int, char>()));
  207. ASSERT_TRUE((cview.empty<const int, const char>()));
  208. const auto e0 = registry.create();
  209. registry.emplace<char>(e0);
  210. const auto e1 = registry.create();
  211. registry.emplace<int>(e1);
  212. ASSERT_FALSE(view.empty());
  213. ASSERT_FALSE((view.empty<int>()));
  214. ASSERT_FALSE((cview.empty<const char>()));
  215. registry.emplace<char>(e1);
  216. auto it = registry.view<int, char>().begin();
  217. ASSERT_EQ(*it, e1);
  218. ASSERT_EQ(++it, (registry.view<int, char>().end()));
  219. ASSERT_NO_THROW((registry.view<int, char>().begin()++));
  220. ASSERT_NO_THROW((++registry.view<int, char>().begin()));
  221. ASSERT_NE(view.begin(), view.end());
  222. ASSERT_NE(cview.begin(), cview.end());
  223. ASSERT_EQ(view.size(), decltype(view.size()){1});
  224. ASSERT_EQ(view.size<int>(), decltype(view.size()){1});
  225. ASSERT_EQ(cview.size<const char>(), decltype(view.size()){2});
  226. registry.get<char>(e0) = '1';
  227. registry.get<char>(e1) = '2';
  228. registry.get<int>(e1) = 42;
  229. for(auto entity: view) {
  230. ASSERT_EQ(std::get<0>(cview.get<const int, const char>(entity)), 42);
  231. ASSERT_EQ(std::get<1>(view.get<int, char>(entity)), '2');
  232. ASSERT_EQ(cview.get<const char>(entity), '2');
  233. }
  234. ASSERT_EQ(*(view.data<int>() + 0), e1);
  235. ASSERT_EQ(*(view.data<char>() + 0), e0);
  236. ASSERT_EQ(*(cview.data<const char>() + 1), e1);
  237. ASSERT_EQ(*(view.raw<int>() + 0), 42);
  238. ASSERT_EQ(*(view.raw<char>() + 0), '1');
  239. ASSERT_EQ(*(cview.raw<const char>() + 1), '2');
  240. }
  241. TEST(MultiComponentView, Iterator) {
  242. entt::registry registry;
  243. const auto entity = registry.create();
  244. registry.emplace<int>(entity);
  245. registry.emplace<char>(entity);
  246. const auto view = registry.view<int, char>();
  247. using iterator = typename decltype(view)::iterator;
  248. iterator end{view.begin()};
  249. iterator begin{};
  250. begin = view.end();
  251. std::swap(begin, end);
  252. ASSERT_EQ(begin, view.begin());
  253. ASSERT_EQ(end, view.end());
  254. ASSERT_NE(begin, end);
  255. ASSERT_EQ(begin++, view.begin());
  256. ASSERT_EQ(begin--, view.end());
  257. ASSERT_EQ(++begin, view.end());
  258. ASSERT_EQ(--begin, view.begin());
  259. ASSERT_EQ(*begin, entity);
  260. ASSERT_EQ(*begin.operator->(), entity);
  261. }
  262. TEST(MultiComponentView, Contains) {
  263. entt::registry registry;
  264. const auto e0 = registry.create();
  265. registry.emplace<int>(e0);
  266. registry.emplace<char>(e0);
  267. const auto e1 = registry.create();
  268. registry.emplace<int>(e1);
  269. registry.emplace<char>(e1);
  270. registry.destroy(e0);
  271. auto view = registry.view<int, char>();
  272. ASSERT_FALSE(view.contains(e0));
  273. ASSERT_TRUE(view.contains(e1));
  274. }
  275. TEST(MultiComponentView, Empty) {
  276. entt::registry registry;
  277. const auto e0 = registry.create();
  278. registry.emplace<double>(e0);
  279. registry.emplace<int>(e0);
  280. registry.emplace<float>(e0);
  281. const auto e1 = registry.create();
  282. registry.emplace<char>(e1);
  283. registry.emplace<float>(e1);
  284. auto view = registry.view<char, int, float>();
  285. ASSERT_EQ(view.size(), entt::registry::size_type{1});
  286. ASSERT_EQ(view.begin(), view.end());
  287. }
  288. TEST(MultiComponentView, Proxy) {
  289. entt::registry registry;
  290. const auto e0 = registry.create();
  291. registry.emplace<int>(e0);
  292. registry.emplace<char>(e0);
  293. const auto e1 = registry.create();
  294. registry.emplace<int>(e1);
  295. registry.emplace<char>(e1);
  296. auto view = registry.view<int, char>();
  297. auto cview = std::as_const(registry).view<const int, const char>();
  298. std::size_t cnt = 0;
  299. view.each([&cnt](auto, int &, char &) { ++cnt; });
  300. view.each([&cnt](int &, char &) { ++cnt; });
  301. for(auto &&[entt, iv, cv]: view.proxy()) {
  302. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  303. static_assert(std::is_same_v<decltype(iv), int &>);
  304. static_assert(std::is_same_v<decltype(cv), char &>);
  305. ++cnt;
  306. }
  307. ASSERT_EQ(cnt, std::size_t{6});
  308. cview.each([&cnt](auto, const int &, const char &) { --cnt; });
  309. cview.each([&cnt](const int &, const char &) { --cnt; });
  310. for(auto &&[entt, iv, cv]: cview.proxy()) {
  311. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  312. static_assert(std::is_same_v<decltype(iv), const int &>);
  313. static_assert(std::is_same_v<decltype(cv), const char &>);
  314. --cnt;
  315. }
  316. ASSERT_EQ(cnt, std::size_t{0});
  317. }
  318. TEST(MultiComponentView, EachWithSuggestedType) {
  319. entt::registry registry;
  320. for(auto i = 0; i < 3; ++i) {
  321. const auto entity = registry.create();
  322. registry.emplace<int>(entity, i);
  323. registry.emplace<char>(entity);
  324. }
  325. // makes char a better candidate during iterations
  326. const auto entity = registry.create();
  327. registry.emplace<int>(entity, 99);
  328. registry.view<int, char>().each<int>([value = 2](const auto curr, const auto) mutable {
  329. ASSERT_EQ(curr, value--);
  330. });
  331. registry.sort<int>([](const auto lhs, const auto rhs) {
  332. return lhs < rhs;
  333. });
  334. registry.view<int, char>().each<int>([value = 0](const auto curr, const auto) mutable {
  335. ASSERT_EQ(curr, value++);
  336. });
  337. registry.sort<int>([](const auto lhs, const auto rhs) {
  338. return lhs > rhs;
  339. });
  340. auto value = registry.view<int, char>().size();
  341. for(auto curr: registry.view<int, char>().proxy()) {
  342. ASSERT_EQ(std::get<1>(curr), --value);
  343. }
  344. registry.sort<int>([](const auto lhs, const auto rhs) {
  345. return lhs < rhs;
  346. });
  347. value = {};
  348. for(auto curr: registry.view<int, char>().proxy<int>()) {
  349. ASSERT_EQ(std::get<1>(curr), value++);
  350. }
  351. }
  352. TEST(MultiComponentView, EachWithHoles) {
  353. entt::registry registry;
  354. const auto e0 = registry.create();
  355. const auto e1 = registry.create();
  356. const auto e2 = registry.create();
  357. registry.emplace<char>(e0, '0');
  358. registry.emplace<char>(e1, '1');
  359. registry.emplace<int>(e0, 0);
  360. registry.emplace<int>(e2, 2);
  361. auto view = registry.view<char, int>();
  362. view.each([e0](auto entity, const char &c, const int &i) {
  363. ASSERT_EQ(entity, e0);
  364. ASSERT_EQ(c, '0');
  365. ASSERT_EQ(i, 0);
  366. });
  367. for(auto curr: view.proxy()) {
  368. ASSERT_EQ(std::get<0>(curr), e0);
  369. ASSERT_EQ(std::get<1>(curr), '0');
  370. ASSERT_EQ(std::get<2>(curr), 0);
  371. }
  372. }
  373. TEST(MultiComponentView, ConstNonConstAndAllInBetween) {
  374. entt::registry registry;
  375. auto view = registry.view<int, const char>();
  376. ASSERT_EQ(view.size(), decltype(view.size()){0});
  377. const auto entity = registry.create();
  378. registry.emplace<int>(entity, 0);
  379. registry.emplace<char>(entity, 'c');
  380. ASSERT_EQ(view.size(), decltype(view.size()){1});
  381. static_assert(std::is_same_v<decltype(view.get<int>({})), int &>);
  382. static_assert(std::is_same_v<decltype(view.get<const char>({})), const char &>);
  383. static_assert(std::is_same_v<decltype(view.get<int, const char>({})), std::tuple<int &, const char &>>);
  384. static_assert(std::is_same_v<decltype(view.raw<const char>()), const char *>);
  385. static_assert(std::is_same_v<decltype(view.raw<int>()), int *>);
  386. view.each([](auto &&i, auto &&c) {
  387. static_assert(std::is_same_v<decltype(i), int &>);
  388. static_assert(std::is_same_v<decltype(c), const char &>);
  389. });
  390. for(auto &&[entt, iv, cv]: view.proxy()) {
  391. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  392. static_assert(std::is_same_v<decltype(iv), int &>);
  393. static_assert(std::is_same_v<decltype(cv), const char &>);
  394. }
  395. }
  396. TEST(MultiComponentView, Find) {
  397. entt::registry registry;
  398. auto view = registry.view<int, const char>();
  399. const auto e0 = registry.create();
  400. registry.emplace<int>(e0);
  401. registry.emplace<char>(e0);
  402. const auto e1 = registry.create();
  403. registry.emplace<int>(e1);
  404. registry.emplace<char>(e1);
  405. const auto e2 = registry.create();
  406. registry.emplace<int>(e2);
  407. registry.emplace<char>(e2);
  408. const auto e3 = registry.create();
  409. registry.emplace<int>(e3);
  410. registry.emplace<char>(e3);
  411. registry.remove<int>(e1);
  412. ASSERT_NE(view.find(e0), view.end());
  413. ASSERT_EQ(view.find(e1), view.end());
  414. ASSERT_NE(view.find(e2), view.end());
  415. ASSERT_NE(view.find(e3), view.end());
  416. auto it = view.find(e2);
  417. ASSERT_EQ(*it, e2);
  418. ASSERT_EQ(*(++it), e3);
  419. ASSERT_EQ(*(++it), e0);
  420. ASSERT_EQ(++it, view.end());
  421. ASSERT_EQ(++view.find(e0), view.end());
  422. const auto e4 = registry.create();
  423. registry.destroy(e4);
  424. const auto e5 = registry.create();
  425. registry.emplace<int>(e5);
  426. registry.emplace<char>(e5);
  427. ASSERT_NE(view.find(e5), view.end());
  428. ASSERT_EQ(view.find(e4), view.end());
  429. }
  430. TEST(MultiComponentView, ExcludedComponents) {
  431. entt::registry registry;
  432. const auto e0 = registry.create();
  433. registry.emplace<int>(e0, 0);
  434. const auto e1 = registry.create();
  435. registry.emplace<int>(e1, 1);
  436. registry.emplace<char>(e1);
  437. const auto e2 = registry.create();
  438. registry.emplace<int>(e2, 2);
  439. const auto e3 = registry.create();
  440. registry.emplace<int>(e3, 3);
  441. registry.emplace<char>(e3);
  442. const auto view = std::as_const(registry).view<const int>(entt::exclude<char>);
  443. for(const auto entity: view) {
  444. ASSERT_TRUE(entity == e0 || entity == e2);
  445. if(entity == e0) {
  446. ASSERT_EQ(view.get<const int>(e0), 0);
  447. } else if(entity == e2) {
  448. ASSERT_EQ(view.get(e2), 2);
  449. }
  450. }
  451. registry.emplace<char>(e0);
  452. registry.emplace<char>(e2);
  453. registry.remove<char>(e1);
  454. registry.remove<char>(e3);
  455. for(const auto entity: view) {
  456. ASSERT_TRUE(entity == e1 || entity == e3);
  457. if(entity == e1) {
  458. ASSERT_EQ(view.get(e1), 1);
  459. } else if(entity == e3) {
  460. ASSERT_EQ(view.get<const int>(e3), 3);
  461. }
  462. }
  463. }
  464. TEST(MultiComponentView, EmptyTypes) {
  465. entt::registry registry;
  466. const auto entity = registry.create();
  467. registry.emplace<int>(entity);
  468. registry.emplace<char>(entity);
  469. registry.emplace<double>(entity);
  470. registry.emplace<empty_type>(entity);
  471. const auto other = registry.create();
  472. registry.emplace<int>(other);
  473. registry.emplace<char>(other);
  474. registry.view<int, char, empty_type>().each([entity](const auto entt, int, char) {
  475. ASSERT_EQ(entity, entt);
  476. });
  477. for(auto &&[entt, iv, cv]: registry.view<int, char, empty_type>().proxy()) {
  478. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  479. static_assert(std::is_same_v<decltype(iv), int &>);
  480. static_assert(std::is_same_v<decltype(cv), char &>);
  481. ASSERT_EQ(entity, entt);
  482. }
  483. registry.view<int, empty_type, char>().each([check = true](int, char) mutable {
  484. ASSERT_TRUE(check);
  485. check = false;
  486. });
  487. for(auto &&[entt, iv, cv]: registry.view<int, empty_type, char>().proxy()) {
  488. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  489. static_assert(std::is_same_v<decltype(iv), int &>);
  490. static_assert(std::is_same_v<decltype(cv), char &>);
  491. ASSERT_EQ(entity, entt);
  492. }
  493. registry.view<empty_type, int, char>().each([entity](const auto entt, int, char) {
  494. ASSERT_EQ(entity, entt);
  495. });
  496. for(auto &&[entt, iv, cv]: registry.view<empty_type, int, char>().proxy()) {
  497. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  498. static_assert(std::is_same_v<decltype(iv), int &>);
  499. static_assert(std::is_same_v<decltype(cv), char &>);
  500. ASSERT_EQ(entity, entt);
  501. }
  502. registry.view<empty_type, int, char>().each<empty_type>([entity](const auto entt, int, char) {
  503. ASSERT_EQ(entity, entt);
  504. });
  505. for(auto &&[entt, iv, cv]: registry.view<empty_type, int, char>().proxy<empty_type>()) {
  506. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  507. static_assert(std::is_same_v<decltype(iv), int &>);
  508. static_assert(std::is_same_v<decltype(cv), char &>);
  509. ASSERT_EQ(entity, entt);
  510. }
  511. registry.view<int, empty_type, char>().each<empty_type>([check = true](int, char) mutable {
  512. ASSERT_TRUE(check);
  513. check = false;
  514. });
  515. for(auto &&[entt, iv, cv]: registry.view<int, empty_type, char>().proxy<empty_type>()) {
  516. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  517. static_assert(std::is_same_v<decltype(iv), int &>);
  518. static_assert(std::is_same_v<decltype(cv), char &>);
  519. ASSERT_EQ(entity, entt);
  520. }
  521. registry.view<int, char, double>().each([entity](const auto entt, int, char, double) {
  522. ASSERT_EQ(entity, entt);
  523. });
  524. for(auto &&[entt, iv, cv, dv]: registry.view<int, char, double>().proxy()) {
  525. static_assert(std::is_same_v<decltype(entt), entt::entity>);
  526. static_assert(std::is_same_v<decltype(iv), int &>);
  527. static_assert(std::is_same_v<decltype(cv), char &>);
  528. static_assert(std::is_same_v<decltype(dv), double &>);
  529. ASSERT_EQ(entity, entt);
  530. }
  531. }
  532. TEST(MultiComponentView, FrontBack) {
  533. entt::registry registry;
  534. auto view = registry.view<const int, const char>();
  535. ASSERT_EQ(view.front(), static_cast<entt::entity>(entt::null));
  536. ASSERT_EQ(view.back(), static_cast<entt::entity>(entt::null));
  537. const auto e0 = registry.create();
  538. registry.emplace<int>(e0);
  539. registry.emplace<char>(e0);
  540. const auto e1 = registry.create();
  541. registry.emplace<int>(e1);
  542. registry.emplace<char>(e1);
  543. const auto entity = registry.create();
  544. registry.emplace<char>(entity);
  545. ASSERT_EQ(view.front(), e1);
  546. ASSERT_EQ(view.back(), e0);
  547. }
  548. TEST(MultiComponentView, ChunkedEmpty) {
  549. entt::registry registry;
  550. auto view = registry.view<const entt::id_type, const char>();
  551. view.chunked([](auto...) { FAIL(); });
  552. registry.emplace<entt::id_type>(registry.create());
  553. registry.emplace<char>(registry.create());
  554. view.chunked([](auto...) { FAIL(); });
  555. }
  556. TEST(MultiComponentView, ChunkedContiguous) {
  557. entt::registry registry;
  558. auto view = registry.view<const entt::id_type, const char>();
  559. for(auto i = 0; i < 5; ++i) {
  560. const auto entity = registry.create();
  561. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  562. registry.emplace<char>(entity);
  563. }
  564. view.chunked([](auto *entity, auto *id, auto *, auto sz) {
  565. ASSERT_EQ(sz, 5u);
  566. for(decltype(sz) i{}; i < sz; ++i) {
  567. ASSERT_EQ(entt::to_integral(*(entity + i)), *(id + i));
  568. }
  569. });
  570. }
  571. TEST(MultiComponentView, ChunkedSpread) {
  572. entt::registry registry;
  573. auto view = registry.view<const entt::id_type, const char>();
  574. registry.emplace<entt::id_type>(registry.create());
  575. for(auto i = 0; i < 3; ++i) {
  576. const auto entity = registry.create();
  577. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  578. registry.emplace<char>(entity);
  579. }
  580. registry.emplace<char>(registry.create());
  581. for(auto i = 0; i < 3; ++i) {
  582. const auto entity = registry.create();
  583. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  584. registry.emplace<char>(entity);
  585. }
  586. registry.emplace<entt::id_type>(registry.create());
  587. registry.emplace<entt::id_type>(registry.create());
  588. registry.emplace<char>(registry.create());
  589. view.chunked([](auto *entity, auto *id, auto *, auto sz) {
  590. ASSERT_EQ(sz, 3u);
  591. for(decltype(sz) i{}; i < sz; ++i) {
  592. ASSERT_EQ(entt::to_integral(*(entity + i)), *(id + i));
  593. }
  594. });
  595. }
  596. TEST(MultiComponentView, ChunkedWithExcludedComponents) {
  597. entt::registry registry;
  598. auto view = registry.view<const entt::id_type, const char>(entt::exclude<double>);
  599. registry.emplace<entt::id_type>(registry.create());
  600. for(auto i = 0; i < 3; ++i) {
  601. const auto entity = registry.create();
  602. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  603. registry.emplace<char>(entity);
  604. }
  605. registry.emplace<char>(registry.create());
  606. for(auto i = 0; i < 2; ++i) {
  607. const auto entity = registry.create();
  608. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  609. registry.emplace<char>(entity);
  610. registry.emplace<double>(entity);
  611. }
  612. for(auto i = 0; i < 3; ++i) {
  613. const auto entity = registry.create();
  614. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  615. registry.emplace<char>(entity);
  616. }
  617. for(auto i = 0; i < 2; ++i) {
  618. const auto entity = registry.create();
  619. registry.emplace<entt::id_type>(entity, entt::to_integral(entity));
  620. registry.emplace<char>(entity);
  621. registry.emplace<double>(entity);
  622. }
  623. registry.emplace<entt::id_type>(registry.create());
  624. registry.emplace<entt::id_type>(registry.create());
  625. registry.emplace<char>(registry.create());
  626. view.chunked([](auto *entity, auto *id, auto *, auto sz) {
  627. ASSERT_EQ(sz, 3u);
  628. for(decltype(sz) i{}; i < sz; ++i) {
  629. ASSERT_EQ(entt::to_integral(*(entity + i)), *(id + i));
  630. }
  631. });
  632. }