9 #include <boost/geometry.hpp>
10 #include <boost/geometry/algorithms/assign.hpp>
11 #include <boost/geometry/algorithms/centroid.hpp>
12 #include <boost/geometry/geometries/point_xy.hpp>
13 #include <boost/geometry/geometries/polygon.hpp>
25 #include <range/v3/range/conversion.hpp>
26 #include <range/v3/view/transform.hpp>
37 std::vector<DetectedHuman> newPoses =
42 return convertHumanPoseToDetectedHuman(measurements.
detectionTime, humanPose);
47 associateHumans(newPoses);
50 for (
const auto& detectedHuman : newPoses)
52 if (not detectedHuman.associated)
55 trackedHumans.push_back(
57 detectedHuman.detectionTime,
59 .trackingId = detectedHuman.trackingId,
60 .associatedCluster =
nullptr,
75 std::vector<AdvancedCluster> potentialFootprints;
76 std::vector<Cluster> unusedClusters;
77 for (
const auto& cluster : measurements.
clusters)
82 .
center = getClusterCenter(cluster), .cluster = cluster, .associated =
false});
86 unusedClusters.push_back(cluster);
91 const auto sortedDistances = getSortedDistances(trackedHumans, potentialFootprints);
93 for (
auto& posDistance : sortedDistances)
99 if (posDistance.oldHuman->associated || posDistance.newCluster->associated)
104 if (!posDistance.oldHuman->associatedCluster)
106 posDistance.oldHuman->associatedCluster = posDistance.newCluster;
107 posDistance.newCluster->associated =
true;
113 auto& footprint1 = posDistance.oldHuman->associatedCluster->cluster;
114 auto& footprint2 = posDistance.newCluster->cluster;
119 (getClusterCenter(footprint1) + getClusterCenter(footprint2)) / 2;
120 pose.linear() = posDistance.oldHuman->humanFilter.get().pose.linear();
123 posDistance.oldHuman->humanFilter.update(pose, measurements.
detectionTime);
125 posDistance.oldHuman->associatedCluster =
nullptr;
126 posDistance.oldHuman->associated =
true;
127 posDistance.newCluster->associated =
true;
133 for (
auto& human : trackedHumans)
135 if (human.associated)
140 if (human.associatedCluster)
142 auto& footprint = human.associatedCluster;
145 pose.translation() = footprint->center;
146 pose.linear() = human.humanFilter.get().pose.linear();
150 human.associatedCluster =
nullptr;
151 human.associated =
true;
157 for (
auto& cluster : potentialFootprints)
159 if (!cluster.associated)
161 unusedClusters.push_back(cluster.cluster);
164 return unusedClusters;
167 std::vector<human::Human>
170 return trackedHumans |
179 trackedHumans.clear();
183 HumanTracker::convertHumanPoseToDetectedHuman(
const DateTime& time,
186 const std::map<std::string, armem::human::PoseKeypoint>& keypoints = humanPose.
keypoints;
190 Eigen::Vector3f centerPos = Eigen::Vector3f::Zero();
192 for (
const auto& [_,
v] : keypoints)
194 if (
v.positionGlobal.has_value())
196 centerPos +=
v.positionGlobal.value().toEigen();
210 Eigen::Vector3f vec(1, 0, 0);
211 Eigen::Vector3f rot3 = qhead._transformVector(vec);
212 Eigen::Vector2f rot2(rot3.x(), rot3.y());
213 if (rot2.norm() != 0)
216 yaw = atan2(rot2.y(), rot2.x());
223 pose.linear() = Eigen::Rotation2Df(yaw).toRotationMatrix();
228 std::vector<HumanTracker::PosHumanDistance>
229 HumanTracker::getSortedDistances(std::vector<HumanTracker::TrackedHuman>& oldHumans,
230 std::vector<HumanTracker::DetectedHuman>& newHumans)
232 std::vector<PosHumanDistance> posDistances;
234 for (
auto& oldHuman : oldHumans)
236 if (oldHuman.associated)
240 for (
auto& newHuman : newHumans)
242 if (newHuman.associated)
248 posDistances.push_back(
251 (newHuman.pose.translation() - oldHuman.humanFilter.get().pose.translation())
257 std::sort(posDistances.begin(),
259 [](
const PosHumanDistance&
a,
const PosHumanDistance& b) ->
bool
260 { return a.distance < b.distance; });
265 std::vector<HumanTracker::PosLaserDistance>
266 HumanTracker::getSortedDistances(std::vector<TrackedHuman>& oldHumans,
267 std::vector<AdvancedCluster>& newClusters)
269 std::vector<PosLaserDistance> posDistances;
271 for (
auto& oldHuman : oldHumans)
273 for (
auto& newCluster : newClusters)
276 posDistances.push_back(
279 (newCluster.center - oldHuman.humanFilter.get().pose.translation()).norm()});
284 std::sort(posDistances.begin(),
286 [](
const PosLaserDistance&
a,
const PosLaserDistance& b) ->
bool
287 { return a.distance < b.distance; });
293 HumanTracker::associateHumans(std::vector<DetectedHuman>& detectedHumans)
296 for (
auto& oldHuman : trackedHumans)
298 if (oldHuman.associated || not oldHuman.trackingId)
302 for (
auto& newHuman : detectedHumans)
304 if (newHuman.associated || not newHuman.trackingId)
310 if (oldHuman.trackingId.value() == newHuman.trackingId.value())
312 associate(&oldHuman, &newHuman);
318 const auto sortedDistances = getSortedDistances(trackedHumans, detectedHumans);
320 for (
auto& posDistance : sortedDistances)
326 if (posDistance.oldHuman->associated || posDistance.newHuman->associated)
331 associate(posDistance.oldHuman, posDistance.newHuman);
336 HumanTracker::associate(TrackedHuman* trackedHuman, DetectedHuman* detectedHuman)
342 trackedHuman->associated =
true;
343 detectedHuman->associated =
true;
345 trackedHuman->humanFilter.update(detectedHuman->pose, detectedHuman->detectionTime);
346 trackedHuman->trackingId = detectedHuman->trackingId;
352 HumanTracker::getClusterSize(
Cluster cluster)
354 return 2 * cluster.circle.radius;
358 HumanTracker::getClusterCenter(
Cluster cluster)
360 typedef boost::geometry::model::d2::point_xy<double> boost_point;
361 typedef boost::geometry::model::polygon<boost_point> boost_polygon;
364 std::vector<boost_point> points;
365 for (
auto& point : cluster.convexHull)
367 points.push_back(boost_point(point.x(), point.y()));
371 boost_point(cluster.convexHull.front().x(), cluster.convexHull.front().y()));
373 boost::geometry::assign_points(poly, points);
376 boost::geometry::centroid(poly, p);
378 return Eigen::Vector2f{p.x(), p.y()};
382 HumanTracker::prepareTrackedHumans(DateTime time)
385 for (
auto it = trackedHumans.begin(); it != trackedHumans.end();)
390 if (human.humanFilter.detectionAge(time) >= parameters.
maxTrackingAge ||
391 human.confidence <= 0)
393 it = trackedHumans.erase(it);
398 human.associatedCluster =
nullptr;
399 human.associated =
false;
400 human.humanFilter.propagation(time);