From a53a6672b8fc89834b1ec3aa7afae150a617473f Mon Sep 17 00:00:00 2001 From: Dan Goodliffe Date: Sun, 23 Mar 2025 13:00:03 +0000 Subject: Populate typed collection of pointers create and emplace only, so far --- lib/collection.h | 79 +++++++++++++++++++++++++++++++++++++++++++++--- test/test-collection.cpp | 16 +++++++++- 2 files changed, 89 insertions(+), 6 deletions(-) diff --git a/lib/collection.h b/lib/collection.h index c843bad..e68a8db 100644 --- a/lib/collection.h +++ b/lib/collection.h @@ -6,12 +6,13 @@ #include #include -template class Collection { +template class Collection { public: virtual ~Collection() = default; using Object = Ptr::element_type; using Objects = std::vector; + template using OtherObjects = std::vector; Collection & operator=(Objects && other) @@ -34,10 +35,14 @@ public: if constexpr (requires(Ptr ptr) { ptr = std::make_shared(std::forward(params)...); }) { auto obj = std::make_shared(std::forward(params)...); objects.emplace_back(obj); + addOthersType(obj.get()); return obj; } else { - return static_cast(objects.emplace_back(std::make_unique(std::forward(params)...)).get()); + auto obj = static_cast( + objects.emplace_back(std::make_unique(std::forward(params)...)).get()); + addOthersType(obj); + return obj; } } @@ -136,11 +141,75 @@ public: auto emplace(Ptr && ptr) { - return objects.emplace_back(std::move(ptr)); + auto object = objects.emplace_back(std::move(ptr)); + addOthersPtr(object.get()); + return object; } protected: Objects objects; + std::tuple...> otherObjects; + + template + void + addOthersType(T * obj) + { + applyToOthersType( + [](auto & others, auto ptr) { + others.emplace_back(ptr); + }, + obj); + } + + void + addOthersPtr(Object * obj) + { + applyToOthersPtr( + [](auto & others, auto ptr) { + others.emplace_back(ptr); + }, + obj); + } + + template + requires(sizeof...(Others) == 0) + void + applyToOthersType(const auto &, T *) + { + } + + void + applyToOthersPtr(const auto &, Object *) + requires(sizeof...(Others) == 0) + { + } + + template + requires(sizeof...(Others) > 0) + void + applyToOthersType(const auto & func, T * obj) + { + ( + [&]() { + if constexpr (std::is_convertible_v) { + std::invoke(func, std::get>(otherObjects), obj); + } + }(), + ...); + } + + void + applyToOthersPtr(const auto & func, Object * obj) + requires(sizeof...(Others) > 0) + { + ( + [&]() { + if (auto ptr = dynamic_cast(obj)) { + std::invoke(func, std::get>(otherObjects), ptr); + } + }(), + ...); + } template auto @@ -168,5 +237,5 @@ protected: } }; -template using SharedCollection = Collection>; -template using UniqueCollection = Collection>; +template using SharedCollection = Collection, Others...>; +template using UniqueCollection = Collection, Others...>; diff --git a/test/test-collection.cpp b/test/test-collection.cpp index 00298bb..5c67a8c 100644 --- a/test/test-collection.cpp +++ b/test/test-collection.cpp @@ -34,7 +34,7 @@ public: } }; -using TestCollection = SharedCollection; +using TestCollection = SharedCollection; BOOST_TEST_DONT_PRINT_LOG_VALUE(TestCollection::Objects::const_iterator) BOOST_TEST_DONT_PRINT_LOG_VALUE(TestCollection::Objects::const_reverse_iterator) @@ -51,12 +51,24 @@ BOOST_AUTO_TEST_CASE(empty) BOOST_AUTO_TEST_CASE(a_base) { auto b = create(); + BOOST_CHECK_EQUAL(objects.size(), 1); + BOOST_CHECK(std::get>(otherObjects).empty()); BOOST_REQUIRE(apply(&Base::add)); BOOST_CHECK_EQUAL(b->total, 1); const auto i = applyOne(&Base::add); BOOST_CHECK_EQUAL(i, end()); } +BOOST_AUTO_TEST_CASE(emplace_others) +{ + emplace(std::make_shared()); + BOOST_CHECK_EQUAL(objects.size(), 1); + BOOST_CHECK(std::get>(otherObjects).empty()); + emplace(std::make_shared()); + BOOST_CHECK_EQUAL(objects.size(), 2); + BOOST_CHECK_EQUAL(std::get>(otherObjects).size(), 1); +} + BOOST_AUTO_TEST_CASE(a_rbase) { auto b = create(); @@ -69,6 +81,8 @@ BOOST_AUTO_TEST_CASE(a_rbase) BOOST_AUTO_TEST_CASE(a_sub) { auto s = create(); + BOOST_CHECK_EQUAL(objects.size(), 1); + BOOST_CHECK_EQUAL(std::get>(otherObjects).size(), 1); BOOST_REQUIRE(apply(&Base::add)); BOOST_CHECK_EQUAL(s->total, 2); const auto i = applyOne(&Base::add); -- cgit v1.2.3