34 #include <SimoxUtility/algorithm/string/string_tools.h>
52 #include <range/v3/range/conversion.hpp>
53 #include <range/v3/view/filter.hpp>
65 const std::string Component::defaultName =
"costmap_merger";
74 properties.costmapsToMerge,
76 "The costmaps to merge, they are prioritized in the order they are specified, i.e. the "
77 "first costmap will take priority over the second one etc.");
80 properties.outputCostmapName,
"p.outputCostmapName",
"The name of the merged costmap.");
107 const std::vector<armem::MemoryID>& snapshotIDs)
111 const std::vector<armem::MemoryID> relevantCostmaps =
113 ranges::views::filter(
116 std::string name =
id.providerSegmentName +
"/" +
id.entityName;
117 return std::find(properties.costmapsToMerge.cbegin(),
118 properties.costmapsToMerge.cend(),
119 name) != properties.costmapsToMerge.cend();
123 if (relevantCostmaps.empty())
128 ARMARX_INFO <<
"Relevant costmaps changed: " << relevantCostmaps;
137 std::lock_guard g{mergeCostmapMtx};
143 std::vector<Costmap> costmaps;
145 for (
const auto& costmap : properties.costmapsToMerge)
152 .
providerName = splits.front(), .name = splits.back(), .timestamp = timestamp};
153 auto result = costmapReaderPlugin->get().query(query);
158 <<
"` which doesn't exist (yet) (" << result.errorMessage <<
")";
163 costmaps.emplace_back(std::move(*result.costmap));
167 if (costmaps.empty())
176 auto mergedCostmap = [&]()
180 Eigen::AlignedBox2f aabb;
182 for (
const auto&
c : costmaps)
185 Eigen::Vector2f globalCornerA =
c.origin() *
c.getLocalSceneBounds().min;
186 Eigen::Vector2f globalCornerB =
c.origin() *
c.getLocalSceneBounds().max;
188 aabb.extend(globalCornerA);
189 aabb.extend(globalCornerB);
191 minCellSize =
std::min(minCellSize,
c.params().cellSize);
194 ARMARX_VERBOSE <<
"Scene bounds intersections is " << aabb.min() <<
" and "
195 << aabb.max() <<
" with minCellSize as " << minCellSize;
201 Eigen::Vector2f boundingSizes = aabb.sizes();
202 const auto cX =
static_cast<std::size_t
>((boundingSizes.x() / minCellSize) + 1);
203 const auto cY =
static_cast<std::size_t
>((boundingSizes.y() / minCellSize) + 1);
205 Eigen::MatrixXf grid(cX, cY);
214 costmap.getMutableMask()->setOnes();
222 const auto& checkInOrder = [&](
int x,
int y,
const Eigen::Vector2f& position)
224 for (
const auto& other : costmaps)
226 Eigen::AlignedBox2f bounds(other.getLocalSceneBounds().min,
227 other.getLocalSceneBounds().max);
228 const auto otherVertex = other.toVertex(position);
230 if (bounds.contains(position) && other.isValid(otherVertex.index))
232 mergedCostmap.getMutableGrid()(x, y) = *other.value(otherVertex.index);
239 for (
int x = 0; x < mergedCostmap.getGrid().rows(); x++)
241 for (
int y = 0; y < mergedCostmap.getGrid().cols(); y++)
245 mergedCostmap.getMutableMask().value()(x, y) = checkInOrder(x, y, position);
249 costmapWriterPlugin->get().store(
250 mergedCostmap, properties.outputCostmapName,
getName(), timestamp);
266 return Component::defaultName;
272 return Component::defaultName;