runtime_view.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. #include <algorithm>
  2. #include <array>
  3. #include <memory>
  4. #include <tuple>
  5. #include <utility>
  6. #include <gtest/gtest.h>
  7. #include <entt/entity/entity.hpp>
  8. #include <entt/entity/runtime_view.hpp>
  9. #include <entt/entity/storage.hpp>
  10. #include "../../common/linter.hpp"
  11. #include "../../common/pointer_stable.h"
  12. template<typename Type>
  13. struct RuntimeView: testing::Test {
  14. using type = Type;
  15. };
  16. using RuntimeViewTypes = ::testing::Types<entt::runtime_view, entt::const_runtime_view>;
  17. TYPED_TEST_SUITE(RuntimeView, RuntimeViewTypes, );
  18. TYPED_TEST(RuntimeView, Functionalities) {
  19. using runtime_view_type = typename TestFixture::type;
  20. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  21. const std::array entity{entt::entity{1}, entt::entity{3}};
  22. runtime_view_type view{};
  23. ASSERT_FALSE(view);
  24. ASSERT_EQ(view.size_hint(), 0u);
  25. ASSERT_EQ(view.begin(), view.end());
  26. ASSERT_FALSE(view.contains(entity[0u]));
  27. ASSERT_FALSE(view.contains(entity[1u]));
  28. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  29. ASSERT_TRUE(view);
  30. ASSERT_EQ(view.size_hint(), 0u);
  31. std::get<1>(storage).emplace(entity[0u]);
  32. std::get<0>(storage).emplace(entity[1u]);
  33. ASSERT_NE(view.size_hint(), 0u);
  34. std::get<1>(storage).emplace(entity[1u]);
  35. ASSERT_EQ(view.size_hint(), 1u);
  36. auto it = view.begin();
  37. ASSERT_EQ(*it, entity[1u]);
  38. ASSERT_EQ(++it, (view.end()));
  39. ASSERT_NO_THROW((view.begin()++));
  40. ASSERT_NO_THROW((++view.begin()));
  41. ASSERT_NE(view.begin(), view.end());
  42. ASSERT_EQ(view.size_hint(), 1u);
  43. std::get<1>(storage).get(entity[0u]) = '1';
  44. std::get<1>(storage).get(entity[1u]) = '2';
  45. std::get<0>(storage).get(entity[1u]) = 3;
  46. for(auto entt: view) {
  47. ASSERT_EQ(std::get<0>(storage).get(entt), 3);
  48. ASSERT_EQ(std::get<1>(storage).get(entt), '2');
  49. }
  50. view.clear();
  51. ASSERT_EQ(view.size_hint(), 0u);
  52. ASSERT_EQ(view.begin(), view.end());
  53. }
  54. TYPED_TEST(RuntimeView, Constructors) {
  55. using runtime_view_type = typename TestFixture::type;
  56. entt::storage<int> storage{};
  57. const entt::entity entity{0};
  58. runtime_view_type view{};
  59. ASSERT_FALSE(view);
  60. storage.emplace(entity);
  61. view = runtime_view_type{std::allocator<int>{}};
  62. view.iterate(storage);
  63. ASSERT_TRUE(view);
  64. ASSERT_TRUE(view.contains(entity));
  65. runtime_view_type temp{view, view.get_allocator()};
  66. const runtime_view_type other{std::move(temp), view.get_allocator()};
  67. test::is_initialized(temp);
  68. ASSERT_FALSE(temp);
  69. ASSERT_TRUE(other);
  70. ASSERT_TRUE(view.contains(entity));
  71. ASSERT_TRUE(other.contains(entity));
  72. }
  73. TYPED_TEST(RuntimeView, Copy) {
  74. using runtime_view_type = typename TestFixture::type;
  75. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  76. const entt::entity entity{0};
  77. runtime_view_type view{};
  78. ASSERT_FALSE(view);
  79. std::get<0>(storage).emplace(entity);
  80. std::get<1>(storage).emplace(entity);
  81. view.iterate(std::get<0>(storage));
  82. runtime_view_type other{view};
  83. ASSERT_TRUE(view);
  84. ASSERT_TRUE(other);
  85. ASSERT_TRUE(view.contains(entity));
  86. ASSERT_TRUE(other.contains(entity));
  87. other.iterate(std::get<0>(storage)).exclude(std::get<1>(storage));
  88. ASSERT_TRUE(view.contains(entity));
  89. ASSERT_FALSE(other.contains(entity));
  90. other = view;
  91. ASSERT_TRUE(view);
  92. ASSERT_TRUE(other);
  93. ASSERT_TRUE(view.contains(entity));
  94. ASSERT_TRUE(other.contains(entity));
  95. }
  96. TYPED_TEST(RuntimeView, Move) {
  97. using runtime_view_type = typename TestFixture::type;
  98. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  99. const entt::entity entity{0};
  100. runtime_view_type view{};
  101. ASSERT_FALSE(view);
  102. std::get<0>(storage).emplace(entity);
  103. std::get<1>(storage).emplace(entity);
  104. view.iterate(std::get<0>(storage));
  105. runtime_view_type other{std::move(view)};
  106. test::is_initialized(view);
  107. ASSERT_FALSE(view);
  108. ASSERT_TRUE(other);
  109. ASSERT_TRUE(other.contains(entity));
  110. view = other;
  111. other.iterate(std::get<0>(storage)).exclude(std::get<1>(storage));
  112. ASSERT_TRUE(view);
  113. ASSERT_TRUE(other);
  114. ASSERT_TRUE(view.contains(entity));
  115. ASSERT_FALSE(other.contains(entity));
  116. other = std::move(view);
  117. test::is_initialized(view);
  118. ASSERT_FALSE(view);
  119. ASSERT_TRUE(other);
  120. ASSERT_TRUE(other.contains(entity));
  121. }
  122. TYPED_TEST(RuntimeView, Swap) {
  123. using runtime_view_type = typename TestFixture::type;
  124. entt::storage<int> storage{};
  125. const entt::entity entity{0};
  126. runtime_view_type view{};
  127. runtime_view_type other{};
  128. ASSERT_FALSE(view);
  129. ASSERT_FALSE(other);
  130. storage.emplace(entity);
  131. view.iterate(storage);
  132. ASSERT_TRUE(view);
  133. ASSERT_FALSE(other);
  134. ASSERT_EQ(view.size_hint(), 1u);
  135. ASSERT_EQ(other.size_hint(), 0u);
  136. ASSERT_TRUE(view.contains(entity));
  137. ASSERT_FALSE(other.contains(entity));
  138. ASSERT_NE(view.begin(), view.end());
  139. ASSERT_EQ(other.begin(), other.end());
  140. view.swap(other);
  141. ASSERT_FALSE(view);
  142. ASSERT_TRUE(other);
  143. ASSERT_EQ(view.size_hint(), 0u);
  144. ASSERT_EQ(other.size_hint(), 1u);
  145. ASSERT_FALSE(view.contains(entity));
  146. ASSERT_TRUE(other.contains(entity));
  147. ASSERT_EQ(view.begin(), view.end());
  148. ASSERT_NE(other.begin(), other.end());
  149. }
  150. TYPED_TEST(RuntimeView, Iterator) {
  151. using runtime_view_type = typename TestFixture::type;
  152. using iterator = typename runtime_view_type::iterator;
  153. entt::storage<int> storage{};
  154. const entt::entity entity{0};
  155. runtime_view_type view{};
  156. storage.emplace(entity);
  157. view.iterate(storage);
  158. iterator end{view.begin()};
  159. iterator begin{};
  160. begin = view.end();
  161. std::swap(begin, end);
  162. ASSERT_EQ(begin, view.begin());
  163. ASSERT_EQ(end, view.end());
  164. ASSERT_NE(begin, end);
  165. ASSERT_EQ(begin++, view.begin());
  166. ASSERT_EQ(begin--, view.end());
  167. ASSERT_EQ(++begin, view.end());
  168. ASSERT_EQ(--begin, view.begin());
  169. ASSERT_EQ(*begin, entity);
  170. ASSERT_EQ(*begin.operator->(), entity);
  171. }
  172. TYPED_TEST(RuntimeView, Contains) {
  173. using runtime_view_type = typename TestFixture::type;
  174. entt::storage<int> storage{};
  175. const std::array entity{entt::entity{1}, entt::entity{3}};
  176. runtime_view_type view{};
  177. storage.emplace(entity[0u]);
  178. storage.emplace(entity[1u]);
  179. storage.erase(entity[0u]);
  180. view.iterate(storage);
  181. ASSERT_FALSE(view.contains(entity[0u]));
  182. ASSERT_TRUE(view.contains(entity[1u]));
  183. }
  184. TYPED_TEST(RuntimeView, Empty) {
  185. using runtime_view_type = typename TestFixture::type;
  186. entt::storage<int> storage{};
  187. const entt::entity entity{0};
  188. runtime_view_type view{};
  189. view.iterate(storage);
  190. ASSERT_FALSE(view.contains(entity));
  191. ASSERT_EQ(view.begin(), view.end());
  192. ASSERT_EQ((std::find(view.begin(), view.end(), entity)), view.end());
  193. storage.emplace(entity);
  194. ASSERT_TRUE(view.contains(entity));
  195. ASSERT_NE(view.begin(), view.end());
  196. ASSERT_NE((std::find(view.begin(), view.end(), entity)), view.end());
  197. }
  198. TYPED_TEST(RuntimeView, Each) {
  199. using runtime_view_type = typename TestFixture::type;
  200. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  201. const std::array entity{entt::entity{1}, entt::entity{3}};
  202. runtime_view_type view{};
  203. std::get<0>(storage).emplace(entity[0u]);
  204. std::get<1>(storage).emplace(entity[0u]);
  205. std::get<1>(storage).emplace(entity[1u]);
  206. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  207. view.each([&](const auto entt) {
  208. ASSERT_EQ(entt, entity[0u]);
  209. });
  210. }
  211. TYPED_TEST(RuntimeView, EachWithHoles) {
  212. using runtime_view_type = typename TestFixture::type;
  213. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  214. const std::array entity{entt::entity{0}, entt::entity{1}, entt::entity{3}};
  215. runtime_view_type view{};
  216. std::get<1>(storage).emplace(entity[0u], '0');
  217. std::get<1>(storage).emplace(entity[1u], '1');
  218. std::get<0>(storage).emplace(entity[0u], 0);
  219. std::get<0>(storage).emplace(entity[2u], 2);
  220. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  221. view.each([&](auto entt) {
  222. ASSERT_EQ(entt, entity[0u]);
  223. });
  224. }
  225. TYPED_TEST(RuntimeView, Exclude) {
  226. using runtime_view_type = typename TestFixture::type;
  227. std::tuple<entt::storage<int>, entt::storage<char>> storage{};
  228. const std::array entity{entt::entity{1}, entt::entity{3}};
  229. runtime_view_type view{};
  230. std::get<0>(storage).emplace(entity[0u]);
  231. std::get<0>(storage).emplace(entity[1u]);
  232. std::get<1>(storage).emplace(entity[1u]);
  233. view.iterate(std::get<0>(storage)).exclude(std::get<1>(storage));
  234. ASSERT_TRUE(view.contains(entity[0u]));
  235. ASSERT_FALSE(view.contains(entity[1u]));
  236. view.each([&](auto entt) {
  237. ASSERT_EQ(entt, entity[0u]);
  238. });
  239. }
  240. TYPED_TEST(RuntimeView, StableType) {
  241. using runtime_view_type = typename TestFixture::type;
  242. std::tuple<entt::storage<int>, entt::storage<test::pointer_stable>> storage{};
  243. const std::array entity{entt::entity{0}, entt::entity{1}, entt::entity{3}};
  244. runtime_view_type view{};
  245. std::get<0>(storage).emplace(entity[0u]);
  246. std::get<0>(storage).emplace(entity[1u]);
  247. std::get<0>(storage).emplace(entity[2u]);
  248. std::get<1>(storage).emplace(entity[0u]);
  249. std::get<1>(storage).emplace(entity[1u]);
  250. std::get<1>(storage).remove(entity[1u]);
  251. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  252. ASSERT_EQ(view.size_hint(), 2u);
  253. ASSERT_TRUE(view.contains(entity[0u]));
  254. ASSERT_FALSE(view.contains(entity[1u]));
  255. ASSERT_EQ(*view.begin(), entity[0u]);
  256. ASSERT_EQ(++view.begin(), view.end());
  257. view.each([&](const auto entt) {
  258. ASSERT_EQ(entt, entity[0u]);
  259. });
  260. for(auto entt: view) {
  261. testing::StaticAssertTypeEq<decltype(entt), entt::entity>();
  262. ASSERT_EQ(entt, entity[0u]);
  263. }
  264. std::get<1>(storage).compact();
  265. ASSERT_EQ(view.size_hint(), 1u);
  266. }
  267. TYPED_TEST(RuntimeView, StableTypeWithExclude) {
  268. using runtime_view_type = typename TestFixture::type;
  269. constexpr entt::entity tombstone = entt::tombstone;
  270. std::tuple<entt::storage<int>, entt::storage<test::pointer_stable>> storage{};
  271. const std::array entity{entt::entity{1}, entt::entity{3}};
  272. runtime_view_type view{};
  273. std::get<1>(storage).emplace(entity[0u], 0);
  274. std::get<1>(storage).emplace(entity[1u], 1);
  275. std::get<0>(storage).emplace(entity[0u]);
  276. view.iterate(std::get<1>(storage)).exclude(std::get<0>(storage));
  277. ASSERT_EQ(view.size_hint(), 2u);
  278. ASSERT_FALSE(view.contains(entity[0u]));
  279. ASSERT_TRUE(view.contains(entity[1u]));
  280. std::get<0>(storage).erase(entity[0u]);
  281. std::get<1>(storage).erase(entity[0u]);
  282. ASSERT_EQ(view.size_hint(), 2u);
  283. ASSERT_FALSE(view.contains(entity[0u]));
  284. ASSERT_TRUE(view.contains(entity[1u]));
  285. for(auto entt: view) {
  286. ASSERT_NE(entt, tombstone);
  287. ASSERT_EQ(entt, entity[1u]);
  288. }
  289. view.each([&](const auto entt) {
  290. ASSERT_NE(entt, tombstone);
  291. ASSERT_EQ(entt, entity[1u]);
  292. });
  293. }
  294. TYPED_TEST(RuntimeView, SameStorageTypes) {
  295. using runtime_view_type = typename TestFixture::type;
  296. std::tuple<entt::storage<int>, entt::storage<int>> storage{};
  297. const std::array entity{entt::entity{1}, entt::entity{3}};
  298. runtime_view_type view{};
  299. std::get<0>(storage).emplace(entity[0u], 2);
  300. std::get<1>(storage).emplace(entity[0u], 3);
  301. std::get<1>(storage).emplace(entity[1u], 1);
  302. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  303. ASSERT_TRUE(view.contains(entity[0u]));
  304. ASSERT_FALSE(view.contains(entity[1u]));
  305. for(auto entt: view) {
  306. ASSERT_EQ(entt, entity[0u]);
  307. }
  308. view.each([&](auto entt) {
  309. ASSERT_EQ(entt, entity[0u]);
  310. });
  311. }
  312. TYPED_TEST(RuntimeView, StorageEntity) {
  313. using runtime_view_type = typename TestFixture::type;
  314. std::tuple<entt::storage<entt::entity>, entt::storage<char>> storage{};
  315. const std::array entity{std::get<0>(storage).generate(), std::get<0>(storage).generate()};
  316. runtime_view_type view{};
  317. std::get<1>(storage).emplace(entity[0u]);
  318. std::get<1>(storage).emplace(entity[1u]);
  319. std::get<1>(storage).erase(entity[0u]);
  320. std::get<0>(storage).erase(entity[0u]);
  321. std::get<0>(storage).bump(entity[0u]);
  322. view.iterate(std::get<0>(storage)).iterate(std::get<1>(storage));
  323. ASSERT_FALSE(view.contains(entity[0u]));
  324. ASSERT_TRUE(view.contains(entity[1u]));
  325. ASSERT_EQ(view.size_hint(), 1u);
  326. ASSERT_NE(view.begin(), view.end());
  327. ASSERT_EQ(std::distance(view.begin(), view.end()), 1);
  328. ASSERT_EQ(*view.begin(), entity[1u]);
  329. for(auto entt: view) {
  330. ASSERT_EQ(entt, entity[1u]);
  331. }
  332. view.each([&entity](auto entt) {
  333. ASSERT_EQ(entt, entity[1u]);
  334. });
  335. }