SimulatedHuman.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  * @author Fabian Reister ( fabian dot reister at kit dot edu )
17  * @date 2022
18  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
19  * GNU General Public License
20  */
21 
22 #include "SimulatedHuman.h"
23 
24 #include <algorithm>
25 #include <optional>
26 
30 
41 
43 {
44 
45 
48  {
49  switch (state_)
50  {
51  case State::Idle:
52  ARMARX_DEBUG << "State:Idle";
53  initialize();
54  state_ = State::Walking;
55  break;
56  case State::Walking:
57  {
58  ARMARX_DEBUG << "State:Walking";
59 
60  step();
61 
62  if (goalReached())
63  {
64  ARMARX_INFO << "Human reached goal";
65  state_ = State::GoalReached;
66  }
67 
68  break;
69  }
70  case State::GoalReached:
71  ARMARX_DEBUG << "State:GoalReached";
72 
73  // TODO(fabian.reister): simulate "perform action at goal": wait a while until movement is started again.
74  state_ = State::Idle;
75  break;
76  }
77 
78  return human_;
79  }
80 
81  bool
83  {
84  const Eigen::Vector2f goal =
85  conv::to2D(globalTrajectory_.points().back().waypoint.pose.translation());
86 
87  return ((goal - human_.pose.translation()).norm() < params_.goalDistanceThreshold);
88  }
89 
90  void
92  {
93  human_.pose = pose;
94 
95  if (goalReached())
96  {
97  ARMARX_VERBOSE << "Goal reached";
98  state_ = State::Idle;
99  }
100 
101  initialize();
102  update();
103  }
104 
105  void
107  {
109  params.linearVelocity = this->params_.maxLinearVelocity;
110  params.algo.obstacleDistanceWeight = 1.1F;
111 
112  core::Scene scene;
113 
114  scene.staticScene.emplace(core::StaticScene{nullptr, {}, {}, std::nullopt, {}});
115  scene.staticScene->distanceToObstaclesCostmap.emplace(distanceField_);
116 
117  global_planning::SPFA planner(params, scene);
118 
119  const core::Pose2D start = human_.pose;
120 
121  if (state_ == State::Idle) // human reached goal and needs a new one
122  {
123  while (true)
124  {
125  const auto sampledPose = algorithms::sampleValidPositionInMap(distanceField_);
126  ARMARX_CHECK(sampledPose.has_value());
127  const core::Pose2D& goal = sampledPose.value();
128 
129 
130  const auto plan = planner.plan(conv::to3D(start), conv::to3D(goal));
131 
132  // check if plan could be created. otherwise try another goal
133  if (plan.has_value())
134  {
135  globalTrajectory_ = plan->trajectory;
136 
137  human_ = human::Human{.pose = start,
138  .linearVelocity = Eigen::Vector2f::Zero(),
139  .detectionTime = Clock::Now()};
140  return;
141  }
142  }
143  }
144  else if (state_ == State::Walking) // human is walking and replans to existing goal
145  {
146  const auto goal = conv::to2D(globalTrajectory_.poses().back());
147  const auto plan = planner.plan(conv::to3D(start), conv::to3D(goal));
148 
149  // check if plan could be created. otherwise try another goal
150  if (plan.has_value())
151  {
152  globalTrajectory_ = plan->trajectory;
153 
154  human_ = human::Human{.pose = start,
155  .linearVelocity = Eigen::Vector2f::Zero(),
156  .detectionTime = Clock::Now()};
157  return;
158  }
159  }
160  }
161 
162  void
164  {
165  const auto updateTime = Clock::Now();
166 
167  const auto duration = updateTime - human_.detectionTime;
168 
169  // move according to old state
170  human_.pose.translation() += duration.toSecondsDouble() * human_.linearVelocity;
171 
172  const auto projection =
173  globalTrajectory_.getProjection(conv::to3D(human_.pose.translation()),
175 
176  human_.pose = conv::to2D(projection.projection.waypoint.pose);
177 
178  const auto wpBefore = projection.wayPointBefore.waypoint.pose.translation();
179  const auto wpAfter = projection.wayPointAfter.waypoint.pose.translation();
180 
181  human_.linearVelocity =
182  (wpAfter - wpBefore).head<2>().normalized() * projection.projection.velocity;
183 
184  human_.linearVelocity =
185  human_.linearVelocity.normalized() * std::clamp(human_.linearVelocity.norm(),
186  params_.minLinearVelocity,
187  params_.maxLinearVelocity);
188 
189  human_.detectionTime = updateTime;
190  }
191 
192  SimulatedHuman::SimulatedHuman(const algorithms::Costmap& distanceField, const Params& params) :
193  distanceField_(distanceField), params_(params)
194  {
195  const auto sampledPose = algorithms::sampleValidPositionInMap(distanceField_);
196  ARMARX_CHECK(sampledPose.has_value());
197  human_.pose = sampledPose.value();
198 
199  update(); // should bring the robot into walking state
200  }
201 } // namespace armarx::navigation::human::simulation
armarx::navigation::global_planning::SPFA
Definition: SPFA.h:69
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
SPFA.h
armarx::navigation::human::Human
Definition: types.h:36
armarx::navigation::core::Scene::staticScene
std::optional< core::StaticScene > staticScene
Definition: types.h:64
armarx::navigation::human::simulation::SimulatedHuman::step
void step()
Definition: SimulatedHuman.cpp:163
armarx::navigation::core::VelocityInterpolation::LinearInterpolation
@ LinearInterpolation
util.h
basic_types.h
armarx::navigation::algorithms::spfa::ShortestPathFasterAlgorithm::Parameters::obstacleDistanceWeight
float obstacleDistanceWeight
Definition: ShortestPathFasterAlgorithm.h:46
armarx::navigation::human::Human::pose
core::Pose2D pose
Definition: types.h:38
types.h
StaticScene.h
armarx::navigation::human::simulation::SimulatedHumanParams::goalDistanceThreshold
float goalDistanceThreshold
Definition: SimulatedHuman.h:34
armarx::navigation::human::simulation::SimulatedHuman::goalReached
bool goalReached() const
Definition: SimulatedHuman.cpp:82
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
clamp
double clamp(double x, double a, double b)
Definition: point.hpp:136
armarx::navigation::core::Pose2D
Eigen::Isometry2f Pose2D
Definition: basic_types.h:34
armarx::navigation::human::Human::detectionTime
DateTime detectionTime
Definition: types.h:40
Clock.h
ShortestPathFasterAlgorithm.h
Costmap.h
armarx::navigation::human::simulation::SimulatedHuman::reinitialize
void reinitialize(const core::Pose2D &pose)
resets the human to the given pose and replans the global optimal trajectory
Definition: SimulatedHuman.cpp:91
armarx::navigation::human::simulation::SimulatedHumanParams::minLinearVelocity
float minLinearVelocity
Definition: SimulatedHuman.h:36
ARMARX_DEBUG
#define ARMARX_DEBUG
Definition: Logging.h:184
armarx::navigation::human::simulation::SimulatedHuman::initialize
void initialize()
Definition: SimulatedHuman.cpp:106
armarx::navigation::human::simulation
This file is part of ArmarX.
Definition: SimulatedHuman.cpp:42
armarx::navigation::human::simulation::SimulatedHuman::SimulatedHuman
SimulatedHuman(const algorithms::Costmap &distanceField, const Params &params=Params())
Definition: SimulatedHuman.cpp:192
armarx::navigation::human::simulation::SimulatedHumanParams
Definition: SimulatedHuman.h:32
armarx::navigation::human::simulation::SimulatedHuman::State::GoalReached
@ GoalReached
armarx::navigation::core::GlobalTrajectory::poses
std::vector< Pose > poses() const noexcept
Definition: Trajectory.cpp:346
armarx::navigation::human::Human::linearVelocity
Eigen::Vector2f linearVelocity
Definition: types.h:39
armarx::navigation::global_planning::SPFAParams::algo
algorithms::spfa::ShortestPathFasterAlgorithm::Parameters algo
Definition: SPFA.h:53
armarx::navigation::core::Scene
Definition: types.h:60
armarx::navigation::human::simulation::SimulatedHuman::update
Human update()
Definition: SimulatedHuman.cpp:47
armarx::navigation::conv::to2D
std::vector< Eigen::Vector2f > to2D(const std::vector< Eigen::Vector3f > &v)
Definition: eigen.cpp:29
ExpressionException.h
armarx::navigation::human::simulation::SimulatedHumanParams::maxLinearVelocity
float maxLinearVelocity
Definition: SimulatedHuman.h:37
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::navigation::algorithms::sampleValidPositionInMap
std::optional< core::Pose2D > sampleValidPositionInMap(const algorithms::Costmap &costmap)
Definition: util.cpp:408
armarx::navigation::global_planning::SPFAParams::linearVelocity
float linearVelocity
Definition: SPFA.h:43
armarx::navigation::conv::to3D
std::vector< Eigen::Vector3f > to3D(const std::vector< Eigen::Vector2f > &v)
Definition: eigen.cpp:14
armarx::navigation::human::simulation::SimulatedHuman::State::Idle
@ Idle
armarx::core::time::Clock::Now
static DateTime Now()
Current time on the virtual clock.
Definition: Clock.cpp:93
SimulatedHuman.h
armarx::navigation::global_planning::SPFA::plan
std::optional< GlobalPlannerResult > plan(const core::Pose &goal) override
Definition: SPFA.cpp:82
armarx::navigation::core::GlobalTrajectory::getProjection
Projection getProjection(const Position &point, const VelocityInterpolation &velocityInterpolation) const
Definition: Trajectory.cpp:245
eigen.h
Logging.h
armarx::navigation::core::StaticScene
Definition: StaticScene.h:43
armarx::navigation::human::simulation::SimulatedHuman::State::Walking
@ Walking
Trajectory.h
armarx::navigation::core::GlobalTrajectory::points
const std::vector< GlobalTrajectoryPoint > & points() const
Definition: Trajectory.cpp:734
types.h
armarx::navigation::global_planning::SPFAParams
Parameters for AStar.
Definition: SPFA.h:41
armarx::navigation::algorithms::Costmap
Definition: Costmap.h:16