From 8734626b2c6fc36acc59e0d7bf7e51d4cff719de Mon Sep 17 00:00:00 2001 From: Dan Goodliffe Date: Sun, 23 Mar 2025 18:09:16 +0000 Subject: Use typed collections for apply/applyOne --- lib/collection.h | 53 ++++++++++++++++++++++++++++++++---------------- test/test-collection.cpp | 28 +++++++++++++++++++++++++ 2 files changed, 64 insertions(+), 17 deletions(-) diff --git a/lib/collection.h b/lib/collection.h index 6ee6c82..d32cff1 100644 --- a/lib/collection.h +++ b/lib/collection.h @@ -86,28 +86,32 @@ public: auto apply(const auto & m, Params &&... params) const { - return apply_internal(objects.begin(), objects.end(), m, std::forward(params)...); + const auto & srcObjects = containerFor(); + return apply_internal(srcObjects.begin(), srcObjects.end(), m, std::forward(params)...); } template auto rapply(const auto & m, Params &&... params) const { - return apply_internal(objects.rbegin(), objects.rend(), m, std::forward(params)...); + const auto & srcObjects = containerFor(); + return apply_internal(srcObjects.rbegin(), srcObjects.rend(), m, std::forward(params)...); } template auto applyOne(const auto & m, Params &&... params) const { - return applyOne_internal(objects.begin(), objects.end(), m, std::forward(params)...); + const auto & srcObjects = containerFor(); + return applyOne_internal(srcObjects.begin(), srcObjects.end(), m, std::forward(params)...); } template auto rapplyOne(const auto & m, Params &&... params) const { - return applyOne_internal(objects.rbegin(), objects.rend(), m, std::forward(params)...); + const auto & srcObjects = containerFor(); + return applyOne_internal(srcObjects.rbegin(), srcObjects.rend(), m, std::forward(params)...); } template @@ -248,25 +252,40 @@ protected: auto apply_internal(const auto begin, const auto end, const auto & m, Params &&... params) const { - return std::count_if(begin, end, [&m, ¶ms...](auto && op) { - if (auto o = dynamic_cast(op.get())) { - std::invoke(m, o, std::forward(params)...); - return true; - } - return false; - }); + if constexpr (std::is_convertible_v) { + std::for_each(begin, end, [&m, ¶ms...](auto && op) { + std::invoke(m, op, std::forward(params)...); + }); + return std::distance(begin, end); + } + else { + return std::count_if(begin, end, [&m, ¶ms...](auto && op) { + if (auto o = dynamic_cast(op.get())) { + std::invoke(m, o, std::forward(params)...); + return true; + } + return false; + }); + } } template auto applyOne_internal(const auto begin, const auto end, const auto & m, Params &&... params) const { - return std::find_if(begin, end, [&m, ¶ms...](auto && op) { - if (auto o = dynamic_cast(op.get())) { - return std::invoke(m, o, std::forward(params)...); - } - return false; - }); + if constexpr (std::is_convertible_v) { + return std::find_if(begin, end, [&m, ¶ms...](auto && op) { + return std::invoke(m, op, std::forward(params)...); + }); + } + else { + return std::find_if(begin, end, [&m, ¶ms...](auto && op) { + if (auto o = dynamic_cast(op.get())) { + return std::invoke(m, o, std::forward(params)...); + } + return false; + }); + } } }; diff --git a/test/test-collection.cpp b/test/test-collection.cpp index 7fadcf9..5aae9f0 100644 --- a/test/test-collection.cpp +++ b/test/test-collection.cpp @@ -21,6 +21,12 @@ public: return false; } + [[nodiscard]] virtual bool + yes() const + { + return true; + } + unsigned int total {0}; }; @@ -137,6 +143,8 @@ BOOST_AUTO_TEST_SUITE_END() using TestUniqueCollection = UniqueCollection; BOOST_TEST_DONT_PRINT_LOG_VALUE(TestUniqueCollection::Objects::const_iterator) BOOST_TEST_DONT_PRINT_LOG_VALUE(TestUniqueCollection::Objects::const_reverse_iterator) +BOOST_TEST_DONT_PRINT_LOG_VALUE(TestUniqueCollection::Objects::iterator) +BOOST_TEST_DONT_PRINT_LOG_VALUE(TestUniqueCollection::Objects::reverse_iterator) BOOST_FIXTURE_TEST_SUITE(utc, TestUniqueCollection) @@ -204,6 +212,26 @@ BOOST_AUTO_TEST_CASE(no_others) emplace(std::make_unique()); } +BOOST_AUTO_TEST_CASE(applyAll) +{ + create(); + BOOST_CHECK_EQUAL(0, apply(&Base::add)); + BOOST_CHECK_EQUAL(1, apply(&Base::add)); + create(); + BOOST_CHECK_EQUAL(1, apply(&Base::add)); + BOOST_CHECK_EQUAL(2, apply(&Base::add)); +} + +BOOST_AUTO_TEST_CASE(applyOneType) +{ + create(); + BOOST_CHECK_EQUAL(objects.end(), applyOne(&Base::yes)); + BOOST_CHECK_EQUAL(objects.begin(), applyOne(&Base::yes)); + create(); + BOOST_CHECK_EQUAL(objects.begin() + 1, applyOne(&Base::yes)); + BOOST_CHECK_EQUAL(objects.begin(), applyOne(&Base::yes)); +} + BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_CASE(wrapped_ptr_file_cons) -- cgit v1.2.3