17 #include "nav2_behaviors/plugins/assisted_teleop.hpp"
18 #include "nav2_util/node_utils.hpp"
20 namespace nav2_behaviors
22 AssistedTeleop::AssistedTeleop()
23 : TimedBehavior<AssistedTeleopAction>(),
24 feedback_(std::make_shared<AssistedTeleopAction::Feedback>())
27 void AssistedTeleop::onConfigure()
29 auto node = node_.lock();
31 throw std::runtime_error{
"Failed to lock node"};
35 nav2_util::declare_parameter_if_not_declared(
37 "projection_time", rclcpp::ParameterValue(1.0));
39 nav2_util::declare_parameter_if_not_declared(
41 "simulation_time_step", rclcpp::ParameterValue(0.1));
43 nav2_util::declare_parameter_if_not_declared(
45 "cmd_vel_teleop", rclcpp::ParameterValue(std::string(
"cmd_vel_teleop")));
47 node->get_parameter(
"projection_time", projection_time_);
48 node->get_parameter(
"simulation_time_step", simulation_time_step_);
50 std::string cmd_vel_teleop;
51 node->get_parameter(
"cmd_vel_teleop", cmd_vel_teleop);
53 vel_sub_ = std::make_unique<nav2_util::TwistSubscriber>(
55 cmd_vel_teleop, rclcpp::SystemDefaultsQoS(),
56 [&](geometry_msgs::msg::Twist::SharedPtr msg) {
57 teleop_twist_.twist = *msg;
58 }, [&](geometry_msgs::msg::TwistStamped::SharedPtr msg) {
62 preempt_teleop_sub_ = node->create_subscription<std_msgs::msg::Empty>(
63 "preempt_teleop", rclcpp::SystemDefaultsQoS(),
65 &AssistedTeleop::preemptTeleopCallback,
66 this, std::placeholders::_1));
69 ResultStatus AssistedTeleop::onRun(
const std::shared_ptr<const AssistedTeleopAction::Goal> command)
71 preempt_teleop_ =
false;
72 command_time_allowance_ = command->time_allowance;
73 end_time_ = this->clock_->now() + command_time_allowance_;
74 return ResultStatus{Status::SUCCEEDED, AssistedTeleopActionResult::NONE,
""};
77 void AssistedTeleop::onActionCompletion(std::shared_ptr<AssistedTeleopActionResult>)
79 teleop_twist_ = geometry_msgs::msg::TwistStamped();
80 preempt_teleop_ =
false;
85 feedback_->current_teleop_duration = elapsed_time_;
86 action_server_->publish_feedback(feedback_);
88 rclcpp::Duration time_remaining = end_time_ - this->clock_->now();
89 if (time_remaining.seconds() < 0.0 && command_time_allowance_.seconds() > 0.0) {
91 std::string error_msg =
"Exceeded time allowance before reaching the " + behavior_name_ +
92 "goal - Exiting " + behavior_name_;
93 RCLCPP_WARN_STREAM(logger_, error_msg.c_str());
94 return ResultStatus{Status::FAILED, AssistedTeleopActionResult::TIMEOUT, error_msg};
98 if (preempt_teleop_) {
100 return ResultStatus{Status::SUCCEEDED, AssistedTeleopActionResult::NONE,
""};
103 geometry_msgs::msg::PoseStamped current_pose;
104 if (!nav2_util::getCurrentPose(
105 current_pose, *tf_, local_frame_, robot_base_frame_,
106 transform_tolerance_))
108 std::string error_msg =
"Current robot pose is not available for " + behavior_name_;
109 RCLCPP_ERROR_STREAM(logger_, error_msg.c_str());
110 return ResultStatus{Status::FAILED, AssistedTeleopActionResult::TF_ERROR, error_msg};
113 geometry_msgs::msg::Pose2D projected_pose;
114 projected_pose.x = current_pose.pose.position.x;
115 projected_pose.y = current_pose.pose.position.y;
116 projected_pose.theta = tf2::getYaw(current_pose.pose.orientation);
118 auto scaled_twist = std::make_unique<geometry_msgs::msg::TwistStamped>(teleop_twist_);
119 for (
double time = simulation_time_step_; time < projection_time_;
120 time += simulation_time_step_)
122 projected_pose = projectPose(projected_pose, teleop_twist_.twist, simulation_time_step_);
124 if (!local_collision_checker_->isCollisionFree(projected_pose)) {
125 if (time == simulation_time_step_) {
126 RCLCPP_DEBUG_STREAM_THROTTLE(
130 behavior_name_.c_str() <<
" collided on first time step, setting velocity to zero");
131 scaled_twist->twist.linear.x = 0.0f;
132 scaled_twist->twist.linear.y = 0.0f;
133 scaled_twist->twist.angular.z = 0.0f;
136 RCLCPP_DEBUG_STREAM_THROTTLE(
140 behavior_name_.c_str() <<
" collision approaching in " << time <<
" seconds");
141 double scale_factor = time / projection_time_;
142 scaled_twist->twist.linear.x *= scale_factor;
143 scaled_twist->twist.linear.y *= scale_factor;
144 scaled_twist->twist.angular.z *= scale_factor;
149 vel_pub_->publish(std::move(scaled_twist));
151 return ResultStatus{Status::RUNNING, AssistedTeleopActionResult::NONE,
""};
154 geometry_msgs::msg::Pose2D AssistedTeleop::projectPose(
155 const geometry_msgs::msg::Pose2D & pose,
156 const geometry_msgs::msg::Twist & twist,
157 double projection_time)
159 geometry_msgs::msg::Pose2D projected_pose = pose;
161 projected_pose.x += projection_time * (
162 twist.linear.x * cos(pose.theta) +
163 twist.linear.y * sin(pose.theta));
165 projected_pose.y += projection_time * (
166 twist.linear.x * sin(pose.theta) -
167 twist.linear.y * cos(pose.theta));
169 projected_pose.theta += projection_time * twist.angular.z;
171 return projected_pose;
174 void AssistedTeleop::preemptTeleopCallback(
const std_msgs::msg::Empty::SharedPtr)
176 preempt_teleop_ =
true;
181 #include "pluginlib/class_list_macros.hpp"
An action server behavior for assisted teleop.
Abstract interface for behaviors to adhere to with pluginlib.