/* * Copyright (c) 2013, Willow Garage, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the name of the Willow Garage, Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include namespace robot_costmap_2d { void calculateMinAndMaxDistances(const std::vector& footprint, double& min_dist, double& max_dist) { min_dist = std::numeric_limits::max(); max_dist = 0.0; if (footprint.size() <= 2) { return; } for (unsigned int i = 0; i < footprint.size() - 1; ++i) { // check the distance from the robot center point to the first vertex double vertex_dist = distance(0.0, 0.0, footprint[i].x, footprint[i].y); double edge_dist = distanceToLine(0.0, 0.0, footprint[i].x, footprint[i].y, footprint[i + 1].x, footprint[i + 1].y); min_dist = std::min(min_dist, std::min(vertex_dist, edge_dist)); max_dist = std::max(max_dist, std::max(vertex_dist, edge_dist)); } // we also need to do the last vertex and the first vertex double vertex_dist = distance(0.0, 0.0, footprint.back().x, footprint.back().y); double edge_dist = distanceToLine(0.0, 0.0, footprint.back().x, footprint.back().y, footprint.front().x, footprint.front().y); min_dist = std::min(min_dist, std::min(vertex_dist, edge_dist)); max_dist = std::max(max_dist, std::max(vertex_dist, edge_dist)); } robot_geometry_msgs::Point32 toPoint32(robot_geometry_msgs::Point pt) { robot_geometry_msgs::Point32 point32; point32.x = pt.x; point32.y = pt.y; point32.z = pt.z; return point32; } robot_geometry_msgs::Point toPoint(robot_geometry_msgs::Point32 pt) { robot_geometry_msgs::Point point; point.x = pt.x; point.y = pt.y; point.z = pt.z; return point; } robot_geometry_msgs::Polygon toPolygon(std::vector pts) { robot_geometry_msgs::Polygon polygon; for (int i = 0; i < pts.size(); i++){ polygon.points.push_back(toPoint32(pts[i])); } return polygon; } std::vector toPointVector(robot_geometry_msgs::Polygon polygon) { std::vector pts; for (int i = 0; i < polygon.points.size(); i++) { pts.push_back(toPoint(polygon.points[i])); } return pts; } void transformFootprint(double x, double y, double theta, const std::vector& footprint_spec, std::vector& oriented_footprint) { // build the oriented footprint at a given location oriented_footprint.clear(); double cos_th = cos(theta); double sin_th = sin(theta); for (unsigned int i = 0; i < footprint_spec.size(); ++i) { robot_geometry_msgs::Point new_pt; new_pt.x = x + (footprint_spec[i].x * cos_th - footprint_spec[i].y * sin_th); new_pt.y = y + (footprint_spec[i].x * sin_th + footprint_spec[i].y * cos_th); oriented_footprint.push_back(new_pt); } } void transformFootprint(double x, double y, double theta, const std::vector& footprint_spec, robot_geometry_msgs::PolygonStamped& oriented_footprint) { // build the oriented footprint at a given location oriented_footprint.polygon.points.clear(); double cos_th = cos(theta); double sin_th = sin(theta); for (unsigned int i = 0; i < footprint_spec.size(); ++i) { robot_geometry_msgs::Point32 new_pt; new_pt.x = x + (footprint_spec[i].x * cos_th - footprint_spec[i].y * sin_th); new_pt.y = y + (footprint_spec[i].x * sin_th + footprint_spec[i].y * cos_th); oriented_footprint.polygon.points.push_back(new_pt); } } void padFootprint(std::vector& footprint, double padding) { // pad footprint in place for (unsigned int i = 0; i < footprint.size(); i++) { robot_geometry_msgs::Point& pt = footprint[ i ]; pt.x += sign0(pt.x) * padding; pt.y += sign0(pt.y) * padding; } } std::vector makeFootprintFromRadius(double radius) { std::vector points; // Loop over 16 angles around a circle making a point each time int N = 16; robot_geometry_msgs::Point pt; for (int i = 0; i < N; ++i) { double angle = i * 2 * M_PI / N; pt.x = cos(angle) * radius; pt.y = sin(angle) * radius; points.push_back(pt); } return points; } bool makeFootprintFromString(const std::string& footprint_string, std::vector& footprint) { std::string error; std::vector > vvf = parseVVF(footprint_string, error); if (error != "") { robot::log_error("Error parsing footprint parameter: '%s'\n", error.c_str()); robot::log_error(" Footprint string was '%s'.\n", footprint_string.c_str()); return false; } // convert vvf into points. if (vvf.size() < 3) { robot::log_error("You must specify at least three points for the robot footprint, reverting to previous footprint.\n"); return false; } footprint.reserve(vvf.size()); for (unsigned int i = 0; i < vvf.size(); i++) { if (vvf[ i ].size() == 2) { robot_geometry_msgs::Point point; point.x = vvf[ i ][ 0 ]; point.y = vvf[ i ][ 1 ]; point.z = 0; footprint.push_back(point); } else { robot::log_error("Points in the footprint specification must be pairs of numbers. Found a point with %d numbers.\n", int(vvf[ i ].size())); return false; } } return true; } std::vector makeFootprintFromParams(robot::NodeHandle& nh) { YAML::Node ft = nh.getParamValue("footprint"); std::vector footprint; if (ft && ft.IsSequence()) { for (size_t i = 0; i < ft.size(); ++i) { auto pt = ft[i]; robot_geometry_msgs::Point p; p.x = pt[0].as(); p.y = pt[1].as(); p.z = 0.0; footprint.push_back(p); std::cout << "Pose[" << i << "]" << p.x << ", " << p.y << std::endl; } } return footprint; } // void writeFootprintToParam(ros::NodeHandle& nh, const std::vector& footprint) // { // std::ostringstream oss; // bool first = true; // for (unsigned int i = 0; i < footprint.size(); i++) // { // robot_geometry_msgs::Point p = footprint[ i ]; // if (first) // { // oss << "[[" << p.x << "," << p.y << "]"; // first = false; // } // else // { // oss << ",[" << p.x << "," << p.y << "]"; // } // } // oss << "]"; // nh.setParam("footprint", oss.str().c_str()); // } double getNumberFromXMLRPC(robot_xmlrpcpp::XmlRpcValue& value, const std::string& full_param_name) { // Make sure that the value we're looking at is either a double or an int. if (value.getType() != robot_xmlrpcpp::XmlRpcValue::TypeInt && value.getType() != robot_xmlrpcpp::XmlRpcValue::TypeDouble) { std::string& value_string = value; robot::log_error("Values in the footprint specification (param %s) must be numbers. Found value %s.\n", full_param_name.c_str(), value_string.c_str()); throw std::runtime_error("Values in the footprint specification must be numbers"); } return value.getType() == robot_xmlrpcpp::XmlRpcValue::TypeInt ? (int)(value) : (double)(value); } std::vector makeFootprintFromXMLRPC(robot_xmlrpcpp::XmlRpcValue& footprint_xmlrpc, const std::string& full_param_name) { // Make sure we have an array of at least 3 elements. if (footprint_xmlrpc.getType() != robot_xmlrpcpp::XmlRpcValue::TypeArray || footprint_xmlrpc.size() < 3) { robot::log_error("The footprint must be specified as list of lists on the parameter server, %s was specified as %s\n", full_param_name.c_str(), std::string(footprint_xmlrpc).c_str()); throw std::runtime_error("The footprint must be specified as list of lists on the parameter server with at least " "3 points eg: [[x1, y1], [x2, y2], ..., [xn, yn]]"); } std::vector footprint; robot_geometry_msgs::Point pt; for (int i = 0; i < footprint_xmlrpc.size(); ++i) { // Make sure each element of the list is an array of size 2. (x and y coordinates) robot_xmlrpcpp::XmlRpcValue point = footprint_xmlrpc[ i ]; if (point.getType() != robot_xmlrpcpp::XmlRpcValue::TypeArray || point.size() != 2) { robot::log_error("The footprint (parameter %s) must be specified as list of lists on the parameter server eg: " "[[x1, y1], [x2, y2], ..., [xn, yn]], but this spec is not of that form.\n", full_param_name.c_str()); throw std::runtime_error("The footprint must be specified as list of lists on the parameter server eg: " "[[x1, y1], [x2, y2], ..., [xn, yn]], but this spec is not of that form"); } pt.x = getNumberFromXMLRPC(point[ 0 ], full_param_name); pt.y = getNumberFromXMLRPC(point[ 1 ], full_param_name); footprint.push_back(pt); } return footprint; } } // end namespace robot_costmap_2d