15 #ifndef OPENNAV_DOCKING__UTILS_HPP_
16 #define OPENNAV_DOCKING__UTILS_HPP_
21 #include "yaml-cpp/yaml.h"
22 #include "rclcpp_lifecycle/lifecycle_node.hpp"
23 #include "rclcpp/rclcpp.hpp"
24 #include "nav2_util/geometry_utils.hpp"
25 #include "angles/angles.h"
26 #include "opennav_docking/types.hpp"
27 #include "tf2/utils.h"
32 using rclcpp::ParameterType::PARAMETER_STRING;
33 using rclcpp::ParameterType::PARAMETER_STRING_ARRAY;
34 using rclcpp::ParameterType::PARAMETER_DOUBLE_ARRAY;
35 using nav2_util::geometry_utils::orientationAroundZAxis;
43 inline bool parseDockFile(
44 const std::string & yaml_filepath,
45 const rclcpp_lifecycle::LifecycleNode::SharedPtr & node,
50 yaml_file = YAML::LoadFile(yaml_filepath);
55 if (!yaml_file[
"docks"]) {
58 "Dock database (%s) does not contain 'docks'.", yaml_filepath.c_str());
62 auto yaml_docks = yaml_file[
"docks"];
64 for (
const auto & yaml_dock : yaml_docks) {
65 std::string dock_name = yaml_dock.first.as<std::string>();
66 const YAML::Node & dock_attribs = yaml_dock.second;
68 curr_dock.frame =
"map";
69 if (dock_attribs[
"frame"]) {
70 curr_dock.frame = dock_attribs[
"frame"].as<std::string>();
73 if (!dock_attribs[
"type"]) {
76 "Dock database (%s) entries do not contain 'type'.", yaml_filepath.c_str());
79 curr_dock.type = dock_attribs[
"type"].as<std::string>();
81 if (!dock_attribs[
"pose"]) {
84 "Dock database (%s) entries do not contain 'pose'.", yaml_filepath.c_str());
87 std::vector<double> pose_arr = dock_attribs[
"pose"].as<std::vector<double>>();
88 if (pose_arr.size() != 3u) {
91 "Dock database (%s) entries do not contain pose of size 3.", yaml_filepath.c_str());
94 curr_dock.pose.position.x = pose_arr[0];
95 curr_dock.pose.position.y = pose_arr[1];
96 curr_dock.pose.orientation = orientationAroundZAxis(pose_arr[2]);
98 if (dock_attribs[
"id"]) {
99 curr_dock.id = dock_attribs[
"id"].as<std::string>();
103 dock_db.emplace(dock_name, curr_dock);
115 inline bool parseDockParams(
116 const std::vector<std::string> & docks_param,
117 const rclcpp_lifecycle::LifecycleNode::SharedPtr & node,
121 std::vector<double> pose_arr;
122 for (
const auto & dock_name : docks_param) {
123 if (!node->has_parameter(dock_name +
".frame")) {
124 node->declare_parameter(dock_name +
".frame",
"map");
126 node->get_parameter(dock_name +
".frame", curr_dock.frame);
128 if (!node->has_parameter(dock_name +
".type")) {
129 node->declare_parameter(dock_name +
".type", PARAMETER_STRING);
131 if (!node->get_parameter(dock_name +
".type", curr_dock.type)) {
132 RCLCPP_ERROR(node->get_logger(),
"Dock %s has no dock 'type'.", dock_name.c_str());
137 if (!node->has_parameter(dock_name +
".pose")) {
138 node->declare_parameter(dock_name +
".pose", PARAMETER_DOUBLE_ARRAY);
140 if (!node->get_parameter(dock_name +
".pose", pose_arr) || pose_arr.size() != 3u) {
141 RCLCPP_ERROR(node->get_logger(),
"Dock %s has no valid 'pose'.", dock_name.c_str());
144 curr_dock.pose.position.x = pose_arr[0];
145 curr_dock.pose.position.y = pose_arr[1];
146 curr_dock.pose.orientation = orientationAroundZAxis(pose_arr[2]);
148 if (!node->has_parameter(dock_name +
".id")) {
149 node->declare_parameter(dock_name +
".id",
"");
151 node->get_parameter(dock_name +
".id", curr_dock.id);
154 dock_db.emplace(dock_name, curr_dock);
165 inline geometry_msgs::msg::PoseStamped getDockPoseStamped(
166 const Dock * dock,
const rclcpp::Time & t)
168 geometry_msgs::msg::PoseStamped pose;
169 pose.pose = dock->pose;
170 pose.header.frame_id = dock->frame;
171 pose.header.stamp = t;
175 inline double l2Norm(
const geometry_msgs::msg::Pose & a,
const geometry_msgs::msg::Pose & b)
177 double angle_a = tf2::getYaw(a.orientation);
178 double angle_b = tf2::getYaw(b.orientation);
179 double delta_angle = angles::shortest_angular_distance(angle_a, angle_b);
181 (a.position.x - b.position.x) * (a.position.x - b.position.x) +
182 (a.position.y - b.position.y) * (a.position.y - b.position.y) +
183 delta_angle * delta_angle);