PersonSimulator.cpp
Go to the documentation of this file.
1 /*
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http://www.gnu.org/licenses/>.
15  *
16  * @package VisionX::COMPONENT_NAME_SNAKE::core
17  * @author Peter Albrecht ( usnlf at student dot kit dot edu)
18  * @date 2024
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 #include "PersonSimulator.h"
24 
25 #include <math.h>
26 
27 #include <optional>
28 #include <sstream>
29 #include <utility>
30 
31 #include <Eigen/Geometry>
32 #include <Eigen/src/Core/Matrix.h>
33 
34 #include <SimoxUtility/math/convert/deg_to_rad.h>
35 #include <SimoxUtility/math/convert/rad_to_deg.h>
36 #include <VirtualRobot/VirtualRobot.h>
37 
41 
43 #include <VisionX/libraries/armem_human/aron/FaceRecognition.aron.generated.h>
44 #include <VisionX/libraries/armem_human/aron/HumanPose.aron.generated.h>
45 #include <VisionX/libraries/armem_human/aron/Profile.aron.generated.h>
52 
54 {
56  remote(r), properties(p)
57  {
58  // these are rough approximations
59  generatedPoseParams = {
60  .hipHeight = 1000.F,
61  .pelvisHeight = 1000.F,
62  .shoulderHeight = 1600.F,
63  .headHeight = 1700.F,
64  .earHeight = 1800.F,
65  };
66  }
67 
68  void
69  PersonSimulator::createHumanProfile(const std::string& name)
70  {
73 
74  // we only set the firstName. FaceImages are not required,
75  // as the user manually sets the profile in the face recognition.
76  // therefore, this will not work with a real implementation of face recognition
77  // TODO: implement option to add face images
79  dto.firstName = name;
80 
81  armarx::armem::MemoryID entityId =
84  .withEntityName(name);
85 
87  .entityID = entityId,
88  .instancesData = {dto.toAron()},
89  .referencedTime = armarx::armem::Time::Now(),
90  .confidence = 1.0F,
91  .sentTime = armarx::armem::Time::Now(),
92  .arrivedTime = armarx::armem::Time::Now(),
93  };
94 
95  ARMARX_INFO << "Committing profile with name: " << name;
96  poseWriter.commit(update);
97  }
98 
99  std::optional<armarx::armem::MemoryID>
100  PersonSimulator::findProfileId(std::string& entityId)
101  {
102  armarx::armem::client::Reader profileReader =
104 
107  std::optional<armarx::armem::MemoryID> foundProfileId = std::nullopt;
108 
109  // memory fetch failed
110  if (not queryResult.success)
111  {
112  return std::nullopt;
113  }
114 
115  // find the first matching profile
116  queryResult.memory.forEachInstance(
117  [&entityId, &foundProfileId](armarx::armem::wm::EntityInstance& instance)
118  {
119  if (foundProfileId.has_value())
120  {
121  // we want the first match, and ignore all further matches here.
122  return;
123  }
124 
125  if (instance.id().entityName == entityId)
126  {
127  // match found
128  foundProfileId = instance.id();
129  }
130  });
131 
132  return foundProfileId;
133  }
134 
135  std::string
136  PersonSimulator::increaseAndConvert(unsigned int& counter)
137  {
138  std::stringstream ss;
139  ss << ++counter;
140  return ss.str();
141  }
142 
143  void
144  PersonSimulator::createFaceRecognition(const Eigen::Vector3f& facePosition,
145  std::optional<std::string> entityName)
146  {
147  ARMARX_INFO << "Creating face recognition at cartesian position: " << facePosition.x()
148  << ", " << facePosition.y() << ", " << facePosition.z();
149  armarx::armem::client::Writer poseWriter =
151 
152 
153  std::optional<armarx::armem::MemoryID> profileId = std::nullopt;
154 
155  if (entityName.has_value())
156  {
157  profileId = findProfileId(entityName.value());
158  }
159 
161  bo.framedPosition3D = armarx::FramedPosition(facePosition, "global", properties.robotName);
162  // doesnt make sense, but this isn't set in prod anyways
163  bo.position2D = Eigen::Vector2i::Identity();
164  bo.position3DGlobal = facePosition;
165  // profile id may be nullopt, but this case is communicated to the user
166  bo.profileID = profileId;
167 
168  armarx::human::arondto::FaceRecognition dto;
170 
171  armarx::armem::MemoryID entityId =
173  .withEntityName(entityName.value_or(increaseAndConvert(faceRecognitionIdCounter)));
174 
176  .entityID = entityId,
177  .instancesData = {dto.toAron()},
178  .referencedTime = armarx::armem::Time::Now(),
179  .confidence = 1.0F,
180  .sentTime = armarx::armem::Time::Now(),
181  .arrivedTime = armarx::armem::Time::Now(),
182  };
183 
184  // hack
186  << "Committing face recognition for profile: "
187  << profileId.value_or(armarx::armem::MemoryID().withEntityName("INVALID")).entityName;
188  poseWriter.commit(update);
189  }
190 
192  PersonSimulator::createHumanPose(const std::string& humanId,
193  Eigen::Vector2f meanFloorPosition,
194  float clockwiseOrientationDegrees)
195  {
196  ARMARX_INFO << "Creating human pose at cartesian position: " << meanFloorPosition.x()
197  << ", " << meanFloorPosition.y();
198  armarx::armem::client::Writer poseWriter =
200 
202  pose.humanTrackingId = humanId;
204  pose.cameraFrameName = "global";
206  armarx::FramedOrientation defaultOrientation(
207  Eigen::Quaternionf::UnitRandom(), "global", "Armar7");
208 
210 
211  pose.keypoints =
212  constructDefaultKeypoints(std::move(meanFloorPosition), clockwiseOrientationDegrees);
213 
214  armarx::armem::MemoryID entityId =
215  armarx::human::PoseCoreSegmentID.withProviderSegmentName(properties.poseProviderName)
216  .withEntityName(humanId);
217 
218  armarx::human::arondto::HumanPose dto;
219  armarx::armem::human::toAron(dto, pose);
220 
222  .entityID = entityId,
223  .instancesData = {dto.toAron()},
224  .referencedTime = armarx::armem::Time::Now(),
225  .confidence = 1.0F,
226  .sentTime = armarx::armem::Time::Now(),
227  .arrivedTime = armarx::armem::Time::Now(),
228  };
229 
230  ARMARX_INFO << "Committing pose with HumanID '" << humanId << "'.";
231  poseWriter.commit(update);
232  return pose;
233  }
234 
237  Eigen::Vector2f meanFloorPosition)
238  {
239  // calculate the needed orientation, then call `createHumanPose`
240 
241  // get robot position, calculate difference
242  const auto robotPose =
244  const Eigen::Vector3f robotPosition = robotPose->translation();
245  ARMARX_DEBUG << "Robot Position: " << robotPosition;
246  Eigen::Vector2f robot2d{robotPosition.x(), robotPosition.y()};
247  Eigen::Vector2f diff = robot2d - meanFloorPosition;
248  diff.normalize();
249  Eigen::Vector2f forwards = Eigen::Vector2f::UnitY();
250 
251  // see: Definition 18.23 in the lecture notes of "Lineare Algebra" by Enrico Leuzinger
252  const float dot = diff.dot(forwards);
253  const float cos_angle = dot;
254  float angle = acosf(cos_angle);
255  if (angle <= (M_PI / 2.F))
256  {
257  angle = -1.F * angle;
258  }
259  angle = simox::math::rad_to_deg(angle);
260 
261  ARMARX_DEBUG << "Calculated angle: " << angle;
262 
263  auto pose = createHumanPose(humanId, meanFloorPosition, angle);
264  return pose;
265  }
266 
267  std::map<std::string, armarx::armem::human::PoseKeypoint>
268  PersonSimulator::constructDefaultKeypoints(Eigen::Vector2f meanFloorPosGlobal,
269  float clockwiseOrientationDegrees)
270  {
272  // General idea: we only explicitly set some keypoints to "plausible" values, as to enable orientation detection
273  // -> set all keypoints to the mean floor projection, then manually set keypoints to rotated values
274  // reference: y is forwards, x is left, z is up
275 
276  /// helper to rotate a vector by the vertical axis
277  auto rotateGlobalVector =
278  [&clockwiseOrientationDegrees, &meanFloorPosGlobal](Eigen::Vector3f globalPosition)
279  {
280  // helper
281  const Eigen::Vector3f projectedMean(
282  meanFloorPosGlobal.x(), meanFloorPosGlobal.y(), 0.F);
283  // create angleaxis from floor projection
284  Eigen::Vector3f verticalAxis = Eigen::Vector3f::UnitZ();
285  Eigen::AngleAxisf verticalAxisRotation = Eigen::AngleAxisf(
286  -1.F * simox::math::deg_to_rad(clockwiseOrientationDegrees), verticalAxis);
287 
288  // make position local, rotate and translate back
289  const Eigen::Vector3f localPosition = globalPosition - projectedMean;
290 
291  const Eigen::Vector3f rotatedLocalPos =
292  verticalAxisRotation.toRotationMatrix() * localPosition;
293 
294  const Eigen::Vector3f rotatedGlobalPos = rotatedLocalPos + projectedMean;
295  return rotatedGlobalPos;
296  };
297 
299 
300  /// note: label denotes the real label without the "Left" or "Right" suffix of the k4a model
301  auto setSymmetricJointsAtHeight = [&](std::string label, float height)
302  {
303  // i shall pray for this to work
304  const std::string left_sf = "Left";
305  const std::string right_sf = "Right";
306 
307  Eigen::Vector3f leftPos =
308  Eigen::Vector3f(meanFloorPosGlobal.x() - properties.jointDisplacement,
309  meanFloorPosGlobal.y(),
310  height);
311  Eigen::Vector3f rightPos =
312  Eigen::Vector3f(meanFloorPosGlobal.x() + properties.jointDisplacement,
313  meanFloorPosGlobal.y(),
314  height);
315  leftPos = rotateGlobalVector(leftPos);
316  rightPos = rotateGlobalVector(rightPos);
317  keypoints[label + left_sf] = constructKeyPoint(leftPos, label + left_sf);
318  keypoints[label + right_sf] = constructKeyPoint(rightPos, label + right_sf);
319  };
320 
321  // assumption: position does not have to be rotated
322  auto setSingleJointAtHeight = [&](std::string label, float height)
323  {
324  Eigen::Vector3f pos =
325  Eigen::Vector3f(meanFloorPosGlobal.x(), meanFloorPosGlobal.y(), height);
326  keypoints[label] = constructKeyPoint(pos, label);
327  };
328 
329  // create keypointmap, initialize all to default
330  for (const auto& joint : JointNames.names())
331  {
332  keypoints[joint] = constructKeyPoint(
333  Eigen::Vector3f(meanFloorPosGlobal.x(), meanFloorPosGlobal.y(), 0.F), joint);
334  }
335 
336  // note: these heights are somewhat arbitrary
337 
338  // set shoulders
339  setSymmetricJointsAtHeight("Shoulder", generatedPoseParams.shoulderHeight);
340 
341  // set hips
342  setSymmetricJointsAtHeight("Hip", generatedPoseParams.hipHeight);
343 
344  // set ears
345  setSymmetricJointsAtHeight("Ear", generatedPoseParams.earHeight);
346 
347  // pelvis and head (needed for positioning/face recognition matching)
348  setSingleJointAtHeight("Pelvis", generatedPoseParams.pelvisHeight);
349  setSingleJointAtHeight("Head", generatedPoseParams.headHeight);
350 
351  // to visualize orientation
352  keypoints["WristRight"] = constructKeyPoint(
353  rotateGlobalVector({meanFloorPosGlobal.x() + properties.jointDisplacement,
354  meanFloorPosGlobal.y() + 100,
355  generatedPoseParams.hipHeight}),
356  "WristRight");
357 
358 
359  return keypoints;
360  }
361 
363  PersonSimulator::constructKeyPoint(Eigen::Vector3f globalPos, const std::string& label)
364  {
365  // helper lambdas to avoid typing
366  auto makeDefaultOrientation = [&](std::string frameName)
367  {
369  Eigen::Quaternionf::Identity(), frameName, properties.robotName);
370  };
371  auto makeDefaultPosition = [&](std::string frameName)
372  {
373  Eigen::Vector3f ident = Eigen::Vector3f::Identity();
374  return armarx::FramedPosition(ident, frameName, properties.robotName);
375  };
376 
377 
379  .label = label,
380  .confidence = 1.0F,
381  .positionCamera = makeDefaultPosition("camera"),
382  .orientationCamera = makeDefaultOrientation("camera"),
383  .positionRobot = makeDefaultPosition("robot"),
384  .orientationRobot = makeDefaultOrientation("robot"),
385  .positionGlobal = armarx::FramedPosition(globalPos, "global", properties.robotName),
386  .orientationGlobal = makeDefaultOrientation("global"),
387 
388  };
389  return key;
390  }
391 
392 } // namespace VisionX::person_simulator::core
armarx::armem::robot_state::RobotReader::queryGlobalPose
std::optional< RobotState::Pose > queryGlobalPose(const std::string &robotName, const armem::Time &timestamp) const
Definition: RobotReader.cpp:378
armarx::armem::detail::SuccessHeader::success
bool success
Definition: SuccessHeader.h:19
armarx::armem::client::Reader
Reads data from a memory server.
Definition: Reader.h:23
VisionX::person_simulator::core
Definition: PersonSimulator.cpp:53
VisionX::person_simulator::core::PersonSimulator::createFaceRecognition
void createFaceRecognition(const Eigen::Vector3f &facePosition, std::optional< std::string > entityName)
Definition: PersonSimulator.cpp:144
Reader.h
armarx::armem::client::QueryResult::memory
wm::Memory memory
The slice of the memory that matched the query.
Definition: Query.h:58
armarx::armem::client::Reader::getLatestSnapshotsIn
QueryResult getLatestSnapshotsIn(const MemoryID &id, armem::query::DataMode dataMode=armem::query::DataMode::WithData) const
Get the latest snapshots under the given memory ID.
Definition: Reader.cpp:340
VisionX::person_simulator::core::PersonSimulator::createHumanPoseFacingRobot
armarx::armem::human::HumanPose createHumanPoseFacingRobot(const std::string &humanId, Eigen::Vector2f meanFloorPosition)
Definition: PersonSimulator.cpp:236
armarx::human::FaceRecognitionCoreSegmentID
const armem::MemoryID FaceRecognitionCoreSegmentID
Definition: memory_ids.cpp:30
VisionX::person_simulator::core::PersonSimulator::findProfileId
std::optional< armarx::armem::MemoryID > findProfileId(std::string &entityId)
Definition: PersonSimulator.cpp:100
armarx::armem::wm::EntityInstance
Client-side working entity instance.
Definition: memory_definitions.h:32
VisionX::person_simulator::core::PersonSimulator::Properties::poseProviderName
std::string poseProviderName
Definition: PersonSimulator.h:63
armarx::armem::human::toAron
void toAron(armarx::human::arondto::HumanPose &dto, const HumanPose &bo)
Definition: aron_conversions.cpp:31
armarx::core::time::DateTime::Now
static DateTime Now()
Definition: DateTime.cpp:51
MemoryID.h
armarx::human::pose::model::k4a_bt_body_32
This file is part of ArmarX.
Definition: k4a_bt_body_32.h:31
PersonSimulator.h
armarx::armem::human::FaceRecognition
Definition: types.h:59
armarx::armem::human::PoseKeypoint::label
std::string label
Definition: types.h:19
memory_ids.h
armarx::armem::client::QueryResult
Result of a QueryInput.
Definition: Query.h:50
armarx::armem::client::MemoryNameSystem::getWriter
Writer getWriter(const MemoryID &memoryID)
Get a writer to the given memory name.
Definition: MemoryNameSystem.cpp:283
k4a_bt_body_32.h
VisionX::person_simulator::core::PersonSimulator::Properties
Fixed properties of this skill implementation.
Definition: PersonSimulator.h:56
armarx::armem::human::HumanPose::poseModelId
std::string poseModelId
Definition: types.h:32
VisionX::person_simulator::core::PersonSimulator::Properties::robotName
std::string robotName
Definition: PersonSimulator.h:59
GfxTL::Identity
void Identity(MatrixXX< N, N, T > *a)
Definition: MatrixXX.h:570
armarx::armem::MemoryID::withProviderSegmentName
MemoryID withProviderSegmentName(const std::string &name) const
Definition: MemoryID.cpp:417
armarx::armem::human::PoseKeypoint
Definition: types.h:17
types.h
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
armarx::human::pose::model::k4a_bt_body_32::JointNames
const simox::meta::EnumNames< Joints > JointNames
Names of the joints as defined in the body model.
Definition: k4a_bt_body_32.h:77
PoseSegment.h
M_PI
#define M_PI
Definition: MathTools.h:17
if
if(!yyvaluep)
Definition: Grammar.cpp:645
FramedPose.h
VisionX::person_simulator::core::PersonSimulator::Properties::profileProviderName
std::string profileProviderName
Definition: PersonSimulator.h:62
ARMARX_DEBUG
#define ARMARX_DEBUG
Definition: Logging.h:184
armarx::armem::human::HumanPose::timestamp
DateTime timestamp
Definition: types.h:37
VisionX::person_simulator::core::PersonSimulator::Remote
Remote parameters of this skill implementation.
Definition: PersonSimulator.h:47
armarx::armem::human::HumanPose::cameraFrameName
std::string cameraFrameName
Definition: types.h:38
armarx::armem::server::human::profile::Profile
armarx::human::arondto::Profile Profile
Definition: Segment.cpp:43
VisionX::person_simulator::core::PersonSimulator::Remote::robotReader
armarx::armem::robot_state::VirtualRobotReader & robotReader
Definition: PersonSimulator.h:52
armarx::armem::EntityUpdate
An update of an entity for a specific point in time.
Definition: Commit.h:25
armarx::armem::client::Writer
Helps a memory client sending data to a memory.
Definition: Writer.h:22
VisionX::person_simulator::core::PersonSimulator::Remote::mns
armarx::armem::client::MemoryNameSystem mns
Definition: PersonSimulator.h:50
armarx::armem::client::Writer::commit
CommitResult commit(const Commit &commit) const
Writes a Commit to the memory.
Definition: Writer.cpp:59
armarx::FramedOrientation
The FramedOrientation class.
Definition: FramedPose.h:215
armarx::armem::human::HumanPose::KeyPointMap
std::map< std::string, PoseKeypoint > KeyPointMap
Definition: types.h:34
armarx::armem::base::detail::MemoryItem::id
MemoryID & id()
Definition: MemoryItem.h:25
armarx::armem::human::HumanPose
Definition: types.h:30
armarx::armem::MemoryID::entityName
std::string entityName
Definition: MemoryID.h:53
armarx::armem::server::ltm::util::mongodb::detail::update
bool update(mongocxx::collection &coll, const nlohmann::json &query, const nlohmann::json &update)
Definition: mongodb.cpp:68
armarx::armem::human::HumanPose::humanTrackingId
std::optional< std::string > humanTrackingId
Definition: types.h:36
armarx::human::pose::model::k4a_bt_body_32::ModelId
const std::string ModelId
Definition: k4a_bt_body_32.h:33
armarx::armem::MemoryID::withEntityName
MemoryID withEntityName(const std::string &name) const
Definition: MemoryID.cpp:425
armarx::VariantType::FramedOrientation
const VariantTypeId FramedOrientation
Definition: FramedPose.h:39
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
aron_conversions.h
Segment.h
armarx::human::ProfileCoreSegmentID
const armem::MemoryID ProfileCoreSegmentID
Definition: memory_ids.cpp:33
Segment.h
armarx::armem::base::detail::ForEachEntityInstanceMixin::forEachInstance
bool forEachInstance(InstanceFunctionT &&func)
Definition: iteration_mixins.h:147
F
Definition: ExportDialogControllerTest.cpp:18
dot
double dot(const Point &x, const Point &y)
Definition: point.hpp:57
angle
double angle(const Point &a, const Point &b, const Point &c)
Definition: point.hpp:109
armarx::human::PoseCoreSegmentID
const armem::MemoryID PoseCoreSegmentID
Definition: memory_ids.cpp:38
armarx::armem::human::HumanPose::keypoints
KeyPointMap keypoints
Definition: types.h:35
VisionX::person_simulator::core::PersonSimulator::GeneratedPoseParams::hipHeight
float hipHeight
Definition: PersonSimulator.h:69
armarx::VariantType::FramedPosition
const VariantTypeId FramedPosition
Definition: FramedPose.h:38
armarx::armem::client::MemoryNameSystem::getReader
Reader getReader(const MemoryID &memoryID)
Get a reader to the given memory name.
Definition: MemoryNameSystem.cpp:191
VisionX::person_simulator::core::PersonSimulator::PersonSimulator
PersonSimulator(const Remote &r, const Properties &p)
Constructor of this skill implementation.
Definition: PersonSimulator.cpp:55
VisionX::person_simulator::core::PersonSimulator::createHumanPose
armarx::armem::human::HumanPose createHumanPose(const std::string &humanId, Eigen::Vector2f meanFloorPosition, float clockwiseOrientationDegrees)
return values are copies of the committed poses, needed for creating a pose+facerecognition
Definition: PersonSimulator.cpp:192
armarx::aron::bo
const std::optional< BoT > & bo
Definition: aron_conversions.h:174
VisionX::person_simulator::core::PersonSimulator::createHumanProfile
void createHumanProfile(const std::string &profileName)
Definition: PersonSimulator.cpp:69