Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add functionality to create Convex shapes from vertices #22390

Open
wants to merge 9 commits into
base: master
Choose a base branch
from
15 changes: 15 additions & 0 deletions geometry/optimization/test/vpolytope_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -294,6 +294,21 @@ GTEST_TEST(VPolytopeTest, NonconvexMesh) {
EXPECT_TRUE(V.PointInSet(V.MaybeGetFeasiblePoint().value()));
}

GTEST_TEST(VPolytopeTest, ToShapeConvex) {
Eigen::Matrix<double, 3, 4> vertices;
vertices.col(0) << 0, 0, 0;
vertices.col(1) << 1, 0, 0;
vertices.col(2) << 0, 1, 0;
vertices.col(3) << 0, 0, 1;

VPolytope V(vertices);
const Convex convex = V.ToShapeConvex();

int num_vertices_of_convex = convex.GetConvexHull().num_vertices();

EXPECT_EQ(vertices.cols(), num_vertices_of_convex);
}

GTEST_TEST(VPolytopeTest, UnitBox6DTest) {
VPolytope V = VPolytope::MakeUnitBox(6);
EXPECT_EQ(V.ambient_dimension(), 6);
Expand Down
5 changes: 5 additions & 0 deletions geometry/optimization/vpolytope.cc
Original file line number Diff line number Diff line change
Expand Up @@ -492,6 +492,11 @@ void VPolytope::WriteObj(const std::filesystem::path& filename) const {
file.close();
}

Convex VPolytope::ToShapeConvex() const {
DRAKE_THROW_UNLESS(ambient_dimension() == 3);
return Convex(vertices_);
}

std::unique_ptr<ConvexSet> VPolytope::DoClone() const {
return std::make_unique<VPolytope>(*this);
}
Expand Down
4 changes: 4 additions & 0 deletions geometry/optimization/vpolytope.h
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,10 @@ class VPolytope final : public ConvexSet {
@pre ambient_dimension() == 3. */
void WriteObj(const std::filesystem::path& filename) const;

/** Creates a geometry::Convex shape using the vertices of this VPolytope.
@pre ambient_dimension() == 3. */
Convex ToShapeConvex() const;

/** Computes the volume of this V-Polytope.
@note this function calls qhull to compute the volume. */
using ConvexSet::CalcVolume;
Expand Down
22 changes: 22 additions & 0 deletions geometry/shape_specification.cc
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,19 @@
#include "drake/geometry/proximity/polygon_to_triangle_mesh.h"
#include "drake/geometry/proximity/triangle_surface_mesh.h"

namespace {
std::string PointsToObjString(const Eigen::Matrix3X<double>& points) {
std::string result = "";

for (int i = 0; i < points.cols(); i++) {
result +=
fmt::format("v {} {} {}\n", points(0, i), points(1, i), points(2, i));
}

return result;
}
} // namespace

namespace drake {
namespace geometry {
namespace {
Expand Down Expand Up @@ -132,6 +145,15 @@ Convex::Convex(MeshSource source, double scale)
ThrowForBadScale(scale, "Convex");
}

Convex::Convex(const Eigen::Matrix3X<double>& points,
const std::string& convex_label, double scale)
: Convex(
InMemoryMesh{
.mesh_file =
MemoryFile(PointsToObjString(points), ".obj", convex_label),
},
scale) {}

std::string Convex::filename() const {
if (source_.is_path()) {
return source_.path().string();
Expand Down
17 changes: 17 additions & 0 deletions geometry/shape_specification.h
Original file line number Diff line number Diff line change
Expand Up @@ -286,6 +286,23 @@ class Convex final : public Shape {
@param scale An optional scale to coordinates. */
explicit Convex(MeshSource source, double scale = 1.0);

/** Constructs an in-memory convex shape specification from the given points.

The convex hull is computed from the points provided. The points are
expected to be in the canonical frame of the shape. Optionally uniformly
scaled by the given scale factor.

@param points The points whose convex hull define the shape.
@param convex_label A label for the object. The label is used for warning
and error messages. Otherwise, the label has no other
functional purpose. It must consist of a single line.
@param scale An optional scale to coordinates.

@throws std::exception if convex_label contains newlines.
@throws std::exception if |scale| < 1e-8. */
explicit Convex(const Eigen::Matrix3X<double>& points,
const std::string& convex_label = {}, double scale = 1.0);

~Convex() final;

/** Returns the source for this specification's mesh data. When working with
Expand Down
33 changes: 31 additions & 2 deletions geometry/test/shape_specification_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -662,6 +662,37 @@ GTEST_TEST(ShapeTest, ConvexFromMemory) {
EXPECT_EQ(hull.num_elements(), 6);
}

GTEST_TEST(ShapeTest, ConvexFromVertices) {
// Make sure the convex hull is automatically taken.
Eigen::Matrix<double, 3, 5> points;
points.col(0) << 0, 0, 0;
points.col(1) << 1, 0, 0;
points.col(2) << 0.25, 0.25, 0.25;
points.col(3) << 0, 1, 0;
points.col(4) << 0, 0, 1;

const double scale = 2.0;

const std::string mesh_name = "a_convex";
const Convex convex(points, mesh_name, scale);

EXPECT_EQ(convex.scale(), scale);
EXPECT_EQ(convex.extension(), ".obj");
const MeshSource& source = convex.source();
ASSERT_TRUE(source.is_in_memory());
EXPECT_EQ(source.in_memory().mesh_file.filename_hint(), mesh_name);

const PolygonSurfaceMesh<double>& hull = convex.GetConvexHull();
EXPECT_EQ(hull.num_vertices(), 4);
EXPECT_EQ(hull.num_elements(), 4);

// Confirm that the cntents of the in-memory mesh file are as expected.
EXPECT_EQ(source.in_memory().mesh_file.contents(),
std::string("v 0 0 0\n") + std::string("v 1 0 0\n") +
std::string("v 0.25 0.25 0.25\n") + std::string("v 0 1 0\n") +
std::string("v 0 0 1\n"));
}

GTEST_TEST(ShapeTest, MeshFromMemory) {
// This will get normalized to ".obj".
const std::string mesh_name = "a_mesh.OBJ";
Expand Down Expand Up @@ -760,7 +791,6 @@ TEST_F(OverrideDefaultGeometryTest, UnsupportedGeometry) {
EXPECT_NO_THROW(this->ImplementGeometry(Sphere(0.5), nullptr));
}


GTEST_TEST(ShapeTest, TypeNameAndToString) {
// In-memory mesh we'll use on Convex and Mesh.
const InMemoryMesh in_memory{
Expand Down Expand Up @@ -828,7 +858,6 @@ GTEST_TEST(ShapeTest, TypeNameAndToString) {
EXPECT_EQ(fmt::to_string(base), "Box(width=1.5, depth=2.5, height=3.5)");
}


GTEST_TEST(ShapeTest, Volume) {
EXPECT_NEAR(CalcVolume(Box(1, 2, 3)), 6.0, 1e-14);
EXPECT_NEAR(CalcVolume(Capsule(1.23, 2.4)),
Expand Down