Component.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 navigation::ArmarXObjects::human_simulator
17  * @author Fabian Reister ( fabian dot reister at kit dot edu )
18  * @date 2022
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 
24 #include "Component.h"
25 
26 #include <algorithm>
27 #include <cstddef>
28 #include <cstdlib>
29 #include <string>
30 #include <vector>
31 
32 #include <SimoxUtility/color/Color.h>
33 #include <VirtualRobot/BoundingBox.h>
34 #include <VirtualRobot/CollisionDetection/CollisionModel.h> // IWYU pragma: keep
35 #include <VirtualRobot/SceneObjectSet.h> // IWYU pragma: keep
36 
47 
53 
59 
60 #include <RVOSimulator.h>
61 #include <Vector2.h>
62 
63 // Include headers you only need in function definitions in the .cpp.
64 
65 // #include <Eigen/Core>
66 
67 // #include <SimoxUtility/color/Color.h>
68 
69 
71 {
73  {
74  addPlugin(virtualRobotReaderPlugin);
75  addPlugin(costmapReaderPlugin);
76  addPlugin(humanWriterPlugin);
77  }
78 
79  const std::string Component::defaultName = "human_simulator";
80 
83  {
86 
87  // Publish to a topic (passing the TopicListenerPrx).
88  // def->topic(myTopicListener);
89 
90  // Subscribe to a topic (passing the topic name).
91  // def->topic<PlatformUnitListener>("MyTopic");
92 
93  // Use (and depend on) another component (passing the ComponentInterfacePrx).
94  // def->component(myComponentProxy)
95 
96 
97  // Add a required property. (The component won't start without a value being set.)
98  // def->required(properties.boxLayerName, "p.box.LayerName", "Name of the box layer in ArViz.");
99 
100  // Add an optionalproperty.
101  def->optional(properties.taskPeriodMs, "p.taskPeriodMs", "");
102  def->optional(properties.nHumans, "p.nHumans", "Number of humans to spawn.");
103 
104  def->optional(
105  properties.humanParams.goalDistanceThreshold, "p.human.goalDistanceThreshold", "");
106  def->optional(properties.humanParams.minLinearVelocity, "p.human.maxLinearVelocity", "");
107  def->optional(properties.humanParams.maxLinearVelocity, "p.human.maxLinearVelocity", "");
108 
109  def->optional(properties.objectPoseProviderName, "p.objectPoseProviderName", "");
110 
111  def->required(properties.robotName, "p.robotName", "");
112 
113  return def;
114  }
115 
116  void
118  {
119  // Topics and properties defined above are automagically registered.
120 
121  // Keep debug observer data until calling `sendDebugObserverBatch()`.
122  // (Requies the armarx::DebugObserverComponentPluginUser.)
123  // setDebugObserverBatchModeEnabled(true);
124  }
125 
126  void
127  Component::setPreferredVelocities(RVO::RVOSimulator* simulator,
128  const std::vector<RVO::Vector2>& goals)
129  {
130  /* Set the preferred velocity to be a vector of unit magnitude (speed) in the
131  * direction of the goal. */
132  // #ifdef _OPENMP
133  // #pragma omp parallel for
134  // #endif /* _OPENMP */
135 
136  for (int i = 0; i < static_cast<int>(properties.nHumans); ++i)
137  {
138  // RVO::Vector2 goalVector = goals[i] - simulator->getAgentPosition(i);
139 
140  // if (RVO::absSq(goalVector) > 1.0F)
141  // {
142  // goalVector = RVO::normalize(goalVector);
143  // }
144 
145  const auto agentPosition = simulator->getAgentPosition(i);
146  auto& simulatedHuman = simulatedHumans.at(i);
147 
149  pose.translation().x() = agentPosition.x() * 1000; // [m] -> [mm]
150  pose.translation().y() = agentPosition.y() * 1000; // [m] -> [mm]
151 
152  simulatedHuman.reinitialize(pose);
153 
154  const std::vector<core::Position> trajectoryPoints =
155  simulatedHuman.trajectory().positions();
156 
157  const std::size_t lookAheadGoalIdx = std::min(trajectoryPoints.size() - 1, 5ul);
158  const auto& intermediateGoal = trajectoryPoints.at(lookAheadGoalIdx);
159 
160  const Eigen::Vector2f rvoGoalHelper = intermediateGoal.head<2>() / 1000; // [mm] -> [m]
161 
162  const RVO::Vector2 rvoGoal(rvoGoalHelper.x(), rvoGoalHelper.y());
163 
164  RVO::Vector2 goalVector = rvoGoal - agentPosition;
165 
166  if (RVO::absSq(goalVector) > 1.0F)
167  {
168  goalVector = RVO::normalize(goalVector);
169  }
170 
171  simulator->setAgentPrefVelocity(i, goalVector);
172  }
173  }
174 
175  void
177  {
178  //
179  // Costmaps
180  //
181 
182  ARMARX_INFO << "Querying costmap";
183 
184  const algorithms::Costmap costmap = [&]() -> algorithms::Costmap
185  {
186  while (true)
187  {
188  const auto timestamp = Clock::Now();
189 
190  const memory::client::costmap::Reader::Query costmapQuery{
191  .providerName = "distance_to_obstacle_costmap_provider", // TODO check
192  .name = "distance_to_obstacles",
193  .timestamp = timestamp};
194 
195  const memory::client::costmap::Reader::Result costmapResult =
196  costmapReaderPlugin->get().query(costmapQuery);
197 
198  // if costmap is not available yet, wait
200  {
202  continue;
203  }
204 
205  ARMARX_CHECK_EQUAL(costmapResult.status,
207 
208  ARMARX_TRACE;
209  ARMARX_CHECK(costmapResult.costmap.has_value());
210  return costmapResult.costmap.value();
211  };
212  }();
213 
214  /// create simulation environment
215  {
216  ARMARX_INFO << "Creating sim";
217 
218  simulator.setTimeStep(static_cast<float>(properties.taskPeriodMs) / 1000);
219  simulator.setAgentDefaults(15.0F, 10U, 1.5F, .1F, .4F, .5F);
220 
221  const objpose::ObjectPoseSeq obstacles =
223  properties.objectPoseProviderName);
224 
225  const auto objects = armarx::objpose::util::asSceneObjects(obstacles);
226 
227 
228  ARMARX_VERBOSE << obstacles.size() << " obstacles";
229 
230  {
231  auto layer = arviz.layer("obstacles");
232 
233 
234  for (const auto& obst : objects->getCollisionModels())
235  {
236 
237  ARMARX_INFO << "adding obstacle ...";
238  const VirtualRobot::BoundingBox bbox = obst->getGlobalBoundingBox();
239 
240  Eigen::Vector2f bbMin = bbox.getMin().head<2>();
241  Eigen::Vector2f bbMax = bbox.getMax().head<2>();
242 
243  // add some margin to the obstacles
244  const float margin = 100; // [mm]
245 
246  bbMin.x() -= margin;
247  bbMin.y() -= margin;
248  bbMax.x() += margin;
249  bbMax.y() += margin;
250 
251  ARMARX_VERBOSE << "Bounding box: (" << bbMin.transpose() << "), ("
252  << bbMax.transpose() << ")";
253 
254  {
255  constexpr float zHeight = 3; // [mm]
256 
257  viz::Polygon polygon(std::to_string(layer.size()));
258  polygon.addPoint(Eigen::Vector3f{bbMin.x(), bbMin.y(), zHeight});
259  polygon.addPoint(Eigen::Vector3f{bbMax.x(), bbMin.y(), zHeight});
260  polygon.addPoint(Eigen::Vector3f{bbMax.x(), bbMax.y(), zHeight});
261  polygon.addPoint(Eigen::Vector3f{bbMin.x(), bbMax.y(), zHeight});
262 
263  polygon.color(simox::Color::black());
264  polygon.color(simox::Color::black());
265 
266  layer.add(polygon);
267  }
268 
269 
270  const Eigen::Vector2d min = bbMin.cast<double>() / 1000; // [mm] -> [m]
271  const Eigen::Vector2d max = bbMax.cast<double>() / 1000; // [mm] -> [m]
272 
273  std::vector<RVO::Vector2> obstacle;
274  obstacle.emplace_back(min.x(), min.y());
275  obstacle.emplace_back(max.x(), min.y());
276  obstacle.emplace_back(max.x(), max.y());
277  obstacle.emplace_back(min.x(), max.y());
278 
279  simulator.addObstacle(obstacle);
280  }
281 
282  arviz.commit(layer);
283  }
284 
285  simulator.processObstacles();
286  }
287 
288  /// create humans in simulation
289  {
290  simulatedHumans.clear();
291 
292  for (std::size_t i = 0U; i < properties.nHumans; ++i)
293  {
295  simulatedHumans.emplace_back(costmap);
296 
297  const RVO::Vector2 position =
298  RVO::Vector2(human.human().pose.translation().x() / 1000,
299  human.human().pose.translation().y() / 1000);
300  auto goal = human.trajectory().positions().back() / 1000; // [mm] -> [m]
301 
302  ARMARX_VERBOSE << "Adding agent";
303  // simulator.addAgent(position, simulator.addGoal(hrvo::Vector2(goal.x(), goal.y())));
304  simulator.addAgent(
305  position); // , simulator.addGoal(hrvo::Vector2(goal.x(), goal.y())));
306 
307  goals.emplace_back(goal.x(), goal.y());
308  }
309  }
310 
311  /// load robot and add it to the simulator
312  {
313  robot = virtualRobotReaderPlugin->get().getRobotWaiting(properties.robotName);
314  ARMARX_CHECK(
315  virtualRobotReaderPlugin->get().synchronizeRobot(*robot, armarx::Clock::Now()));
316 
317  const auto robotPosH = robot->getGlobalPosition();
318  const RVO::Vector2 robotPos(robotPosH.x() / 1000, robotPosH.y() / 1000);
319 
320  robotAgentId = simulator.addAgent(robotPos);
321  simulator.setAgentMaxSpeed(robotAgentId,
322  0); // we do not allow the robot to be perturbed
323  simulator.setAgentPrefVelocity(robotAgentId, RVO::Vector2(0, 0));
324  simulator.setAgentRadius(robotAgentId, 0.6);
325  }
326 
327  /// start task
328  task = new PeriodicTask<Component>(
329  this, &Component::runPeriodically, properties.taskPeriodMs, false, "runningTask");
330  task->start();
331  }
332 
333  void
335  {
336  task->stop();
337  }
338 
339  void
341  {
342  }
343 
344  std::string
346  {
347  return Component::defaultName;
348  }
349 
350  std::string
352  {
353  return Component::defaultName;
354  }
355 
356  void
357  Component::runPeriodically()
358  {
359  const DateTime timestamp = Clock::Now();
360 
361  setPreferredVelocities(&simulator, goals);
362 
363  // TODO(fabian.reister): check how much the agents interact => this can be an evaluation criterion
364 
365  // update robot
366  {
367  ARMARX_CHECK(
368  virtualRobotReaderPlugin->get().synchronizeRobot(*robot, armarx::Clock::Now()));
369 
370  const auto robotPosH = robot->getGlobalPosition();
371  const RVO::Vector2 robotPos(robotPosH.x() / 1000, robotPosH.y() / 1000);
372 
373  simulator.setAgentPosition(robotAgentId, robotPos);
374  }
375 
376  simulator.doStep();
377 
378  // visualize paths
379  {
380  auto layer = arviz.layer("trajectories");
381  for (const auto& simulatedHuman : simulatedHumans)
382  {
383  viz::Path path(std::to_string(layer.size()));
384 
385  for (const auto& pt : simulatedHuman.trajectory().points())
386  {
387  path.addPoint(pt.waypoint.pose.translation());
388  }
389 
390  // unique and distinct color for this human
391  path.colorGlasbeyLUT(layer.size());
392 
393  layer.add(path);
394  }
395 
396  arviz.commit(layer);
397  }
398 
399 
400  human::Humans humans;
401  for (std::size_t i = 0; i < properties.nHumans; i++)
402  {
403  const auto pos = simulator.getAgentPosition(i);
404 
405  human::Human human;
406  human.pose.setIdentity();
407  human.pose.translation().x() = pos.x() * 1000; // [m] -> [mm]
408  human.pose.translation().y() = pos.y() * 1000; // [m] -> [mm]
409 
410  humans.push_back(human);
411  }
412 
413  ARMARX_VERBOSE << "Updating humans.";
414  humanWriterPlugin->get().store(humans, getName(), timestamp);
415  }
416 
417  /* (Requires the armarx::LightweightRemoteGuiComponentPluginUser.)
418  void
419  Component::createRemoteGuiTab()
420  {
421  using namespace armarx::RemoteGui::Client;
422 
423  // Setup the widgets.
424 
425  tab.boxLayerName.setValue(properties.boxLayerName);
426 
427  tab.numBoxes.setValue(properties.numBoxes);
428  tab.numBoxes.setRange(0, 100);
429 
430  tab.drawBoxes.setLabel("Draw Boxes");
431 
432  // Setup the layout.
433 
434  GridLayout grid;
435  int row = 0;
436  {
437  grid.add(Label("Box Layer"), {row, 0}).add(tab.boxLayerName, {row, 1});
438  ++row;
439 
440  grid.add(Label("Num Boxes"), {row, 0}).add(tab.numBoxes, {row, 1});
441  ++row;
442 
443  grid.add(tab.drawBoxes, {row, 0}, {2, 1});
444  ++row;
445  }
446 
447  VBoxLayout root = {grid, VSpacer()};
448  RemoteGui_createTab(getName(), root, &tab);
449  }
450 
451 
452  void
453  Component::RemoteGui_update()
454  {
455  if (tab.boxLayerName.hasValueChanged() || tab.numBoxes.hasValueChanged())
456  {
457  std::scoped_lock lock(propertiesMutex);
458  properties.boxLayerName = tab.boxLayerName.getValue();
459  properties.numBoxes = tab.numBoxes.getValue();
460 
461  {
462  setDebugObserverDatafield("numBoxes", properties.numBoxes);
463  setDebugObserverDatafield("boxLayerName", properties.boxLayerName);
464  sendDebugObserverBatch();
465  }
466  }
467  if (tab.drawBoxes.wasClicked())
468  {
469  // Lock shared variables in methods running in seperate threads
470  // and pass them to functions. This way, the called functions do
471  // not need to think about locking.
472  std::scoped_lock lock(propertiesMutex, arvizMutex);
473  drawBoxes(properties, arviz);
474  }
475  }
476  */
477 
478 
479  /* (Requires the armarx::ArVizComponentPluginUser.)
480  void
481  Component::drawBoxes(const Component::Properties& p, viz::Client& arviz)
482  {
483  // Draw something in ArViz (requires the armarx::ArVizComponentPluginUser.
484  // See the ArVizExample in RobotAPI for more examples.
485 
486  viz::Layer layer = arviz.layer(p.boxLayerName);
487  for (int i = 0; i < p.numBoxes; ++i)
488  {
489  layer.add(viz::Box("box_" + std::to_string(i))
490  .position(Eigen::Vector3f(i * 100, 0, 0))
491  .size(20).color(simox::Color::blue()));
492  }
493  arviz.commit(layer);
494  }
495  */
496 
497 
499 
500 } // namespace armarx::navigation::components::human_simulator
armarx::navigation::components::human_simulator::ARMARX_REGISTER_COMPONENT_EXECUTABLE
ARMARX_REGISTER_COMPONENT_EXECUTABLE(Component, Component::GetDefaultName())
armarx::viz::Client::commit
CommitResult commit(StagedCommit const &commit)
Definition: Client.cpp:89
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
Path.h
armarx::objpose::ObjectPoseSeq
std::vector< ObjectPose > ObjectPoseSeq
Definition: forward_declarations.h:20
basic_types.h
ObjectPoseClientPlugin.h
armarx::core::time::Clock::WaitFor
static void WaitFor(const Duration &duration)
Wait for a certain duration on the virtual clock.
Definition: Clock.cpp:99
armarx::navigation::human::simulation::SimulatedHuman::human
human::Human & human()
Definition: SimulatedHuman.h:50
DateTime.h
trace.h
armarx::max
std::vector< T > max(const std::vector< T > &v1, const std::vector< T > &v2)
Definition: VectorHelpers.h:297
Duration.h
Reader.h
PeriodicTask.h
util.h
armarx::navigation::human::Human::pose
core::Pose2D pose
Definition: types.h:38
types.h
armarx::navigation::components::human_simulator::Component::onDisconnectComponent
void onDisconnectComponent() override
Definition: Component.cpp:334
armarx::ManagedIceObject::addPlugin
PluginT * addPlugin(const std::string prefix="", ParamsT &&... params)
Definition: ManagedIceObject.h:186
armarx::VariantType::Vector2
const VariantTypeId Vector2
Definition: Pose.h:37
armarx::navigation::components::human_simulator::Component::getDefaultName
std::string getDefaultName() const override
Definition: Component.cpp:345
armarx::navigation::human::simulation::SimulatedHuman
Definition: SimulatedHuman.h:40
armarx::navigation::human::Humans
std::vector< Human > Humans
Definition: types.h:48
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
Elements.h
armarx::navigation::core::Pose2D
Eigen::Isometry2f Pose2D
Definition: basic_types.h:34
Clock.h
armarx::navigation::memory::client::costmap::Reader::Query::providerName
std::string providerName
Definition: Reader.h:44
Costmap.h
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
GfxTL::Identity
void Identity(MatrixXX< N, N, T > *a)
Definition: MatrixXX.h:570
armarx::navigation::components::human_simulator::Component::GetDefaultName
static std::string GetDefaultName()
Get the component's default name.
Definition: Component.cpp:351
armarx::navigation::memory::client::costmap::Reader::Result::costmap
std::optional< algorithms::Costmap > costmap
Definition: Reader.h:51
armarx::navigation::human::simulation::SimulatedHuman::trajectory
core::GlobalTrajectory & trajectory()
Definition: SimulatedHuman.h:56
armarx::viz::Polygon::addPoint
Polygon & addPoint(Eigen::Vector3f p)
Definition: Elements.h:302
armarx::viz::Polygon
Definition: Elements.h:260
armarx::navigation::memory::client::costmap::Reader::Result::status
enum armarx::navigation::memory::client::costmap::Reader::Result::Status status
armarx::navigation::memory::client::costmap::Reader::Query
Definition: Reader.h:42
armarx::to_string
const std::string & to_string(const std::string &s)
Definition: StringHelpers.h:41
Component.h
armarx::navigation::components::human_simulator::Component::Component
Component()
Definition: Component.cpp:72
ExpressionException.h
armarx::core::time::DateTime
Represents a point in time.
Definition: DateTime.h:24
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
Decoupled.h
armarx::navigation::memory::client::costmap::Reader::Result
Definition: Reader.h:49
armarx::navigation::components::human_simulator::Component::onExitComponent
void onExitComponent() override
Definition: Component.cpp:340
forward_declarations.h
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::navigation::components::human_simulator::Component::onConnectComponent
void onConnectComponent() override
Definition: Component.cpp:176
PropertyDefinitionContainer.h
armarx::ArVizComponentPluginUser::arviz
armarx::viz::Client arviz
Definition: ArVizComponentPlugin.h:42
armarx::objpose::util::asSceneObjects
VirtualRobot::SceneObjectSetPtr asSceneObjects(const objpose::ObjectPoseSeq &objectPoses)
Definition: util.cpp:133
IceUtil::Handle< class PropertyDefinitionContainer >
armarx::min
std::vector< T > min(const std::vector< T > &v1, const std::vector< T > &v2)
Definition: VectorHelpers.h:327
armarx::viz::ElementOps::color
DerivedT & color(Color color)
Definition: ElementOps.h:218
F
Definition: ExportDialogControllerTest.cpp:18
armarx::navigation::core::GlobalTrajectory::positions
std::vector< Position > positions() const noexcept
Definition: Trajectory.cpp:332
armarx::navigation::components::human_simulator::Component::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: Component.cpp:82
armarx::core::time::Clock::Now
static DateTime Now()
Current time on the virtual clock.
Definition: Clock.cpp:93
armarx::ManagedIceObject::getName
std::string getName() const
Retrieve name of object.
Definition: ManagedIceObject.cpp:108
SimulatedHuman.h
armarx::PeriodicTask
Definition: ArmarXManager.h:70
Logging.h
min
T min(T t1, T t2)
Definition: gdiam.h:44
armarx::viz::Path
Definition: Path.h:31
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
armarx::viz::Client::layer
Layer layer(std::string const &name) const
Definition: Client.cpp:80
armarx::navigation::components::human_simulator
Definition: Component.cpp:70
armarx::navigation::components::human_simulator::Component::onInitComponent
void onInitComponent() override
Definition: Component.cpp:117
armarx::navigation::algorithms::Costmap
Definition: Costmap.h:16
armarx::core::time::Duration::MilliSeconds
static Duration MilliSeconds(std::int64_t milliSeconds)
Constructs a duration in milliseconds.
Definition: Duration.cpp:48
armarx::navigation::memory::client::costmap::Reader::Result::Success
@ Success
Definition: Reader.h:55
armarx::ObjectPoseClientPluginUser::getObjectPosesByProvider
objpose::ObjectPoseSeq getObjectPosesByProvider(const std::string &providerName)
Definition: ObjectPoseClientPlugin.cpp:82