#pragma once #include #include #include #include #include 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) { objects = std::move(other); return *this; } const Ptr & operator[](size_t idx) const { return objects[idx]; } template auto create(Params &&... params) requires std::is_base_of_v { 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 { auto obj = static_cast( objects.emplace_back(std::make_unique(std::forward(params)...)).get()); addOthersType(obj); return obj; } } template T * find() { if (auto i = std::find_if(objects.begin(), objects.end(), [](auto && o) { return (dynamic_cast(o.get())); }); i != objects.end()) { return static_cast(i->get()); } return nullptr; } template auto findOrCreate(Params &&... params) requires std::is_base_of_v { if (auto o = find()) { return o; } return create(std::forward(params)...).get(); } template auto apply(const auto & m, Params &&... params) const { return apply_internal(objects.begin(), objects.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)...); } template auto applyOne(const auto & m, Params &&... params) const { return applyOne_internal(objects.begin(), objects.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)...); } template auto removeAll() { return std::erase_if(objects, [](auto && op) { return dynamic_cast(op.get()); }); } [[nodiscard]] auto begin() const { return objects.begin(); } [[nodiscard]] auto end() const { return objects.end(); } [[nodiscard]] auto rbegin() const { return objects.rbegin(); } [[nodiscard]] auto rend() const { return objects.rend(); } [[nodiscard]] bool empty() const { return objects.empty(); } auto emplace(Ptr && 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 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; }); } 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; }); } }; template using SharedCollection = Collection, Others...>; template using UniqueCollection = Collection, Others...>;