Skip to content

Commit

Permalink
Support vertex-based mesh definitions in MJCF
Browse files Browse the repository at this point in the history
  • Loading branch information
JafarAbdi committed Jan 1, 2025
1 parent a73c947 commit f0d5a0e
Show file tree
Hide file tree
Showing 4 changed files with 97 additions and 10 deletions.
2 changes: 2 additions & 0 deletions include/pinocchio/parsers/mjcf/mjcf-graph.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,8 @@ namespace pinocchio
Eigen::Vector3d scale = Eigen::Vector3d::Constant(1);
// Path to the mesh file
std::string filePath;
// Vertices of the mesh
hpp::fcl::MatrixX3s vertices;
};

/// @brief All informations related to a texture are stored here
Expand Down
13 changes: 13 additions & 0 deletions src/parsers/mjcf/mjcf-graph-geom.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,19 @@ namespace pinocchio
if (geom.geomType == "mesh")
{
MjcfMesh currentMesh = currentGraph.mapOfMeshes.at(geom.meshName);
if (currentMesh.vertices.size() > 0)
{
auto vertices = currentMesh.vertices;
// Scale vertices
for (std::size_t i = 0; i < vertices.rows(); ++i)
vertices.row(i) = vertices.row(i).cwiseProduct(currentMesh.scale.transpose());
auto model = std::make_shared<hpp::fcl::BVHModel<fcl::OBBRSS>>();
model->beginModel();
model->addVertices(vertices);
model->endModel();
model->buildConvexHull(true, "Qt");
return model->convex;
}
meshPath = currentMesh.filePath;
meshScale = currentMesh.scale;
hpp::fcl::BVHModelPtr_t bvh = meshLoader->load(meshPath, meshScale);
Expand Down
51 changes: 41 additions & 10 deletions src/parsers/mjcf/mjcf-graph.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -582,20 +582,51 @@ namespace pinocchio

MjcfMesh mesh;
auto file = el.get_optional<std::string>("<xmlattr>.file");
if (!file)
throw std::invalid_argument("Only meshes with files are supported");

fs::path filePath(*file);
std::string name = getName(el, filePath);

mesh.filePath =
updatePath(compilerInfo.strippath, compilerInfo.meshdir, modelPath, filePath).string();

auto scale = el.get_optional<std::string>("<xmlattr>.scale");
if (scale)
mesh.scale = internal::getVectorFromStream<3>(*scale);
if (file)
{
fs::path filePath(*file);
std::string name = getName(el, filePath);

mesh.filePath =
updatePath(compilerInfo.strippath, compilerInfo.meshdir, modelPath, filePath).string();
mapOfMeshes.insert(std::make_pair(name, mesh));
return;
}

// Handle vertex-based mesh
auto vertex = el.get_optional<std::string>("<xmlattr>.vertex");
if (!vertex)
{
throw std::invalid_argument("Only meshes with files/vertices are supported");
}

auto name = el.get_optional<std::string>("<xmlattr>.name");
if (!name)
{
throw std::invalid_argument("Mesh with vertices without a name is not supported");
}

// Parse and validate vertices
Eigen::VectorXd meshVertices = internal::getUnknownSizeVectorFromStream(*vertex);
if (meshVertices.size() % 3 != 0)
{
throw std::invalid_argument("Number of vertices is not a multiple of 3");
}

// Convert to 3D vertex matrix
const size_t numVertices = meshVertices.size() / 3;
hpp::fcl::MatrixX3s vertices(numVertices, 3);

for (size_t i = 0; i < meshVertices.size(); i += 3)
{
vertices.row(i / 3) = meshVertices.segment<3>(i).transpose();
}

mapOfMeshes.insert(std::make_pair(name, mesh));
mesh.vertices = vertices;
mapOfMeshes.insert(std::make_pair(*name, mesh));
}

void MjcfGraph::parseAsset(const ptree & el)
Expand Down
41 changes: 41 additions & 0 deletions unittest/mjcf.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1357,4 +1357,45 @@ BOOST_AUTO_TEST_CASE(test_default_eulerseq)
BOOST_CHECK(graph.mapOfBodies["body"].bodyPlacement.isApprox(placement));
}

/// @brief Test parsing a mesh with vertices
/// @param
BOOST_AUTO_TEST_CASE(parse_mesh_with_vertices)
{
std::istringstream xmlDataNoStrip(R"(<mujoco model="parseVertices">
<asset>
<mesh name="chasis" scale=".01 .006 .0015"
vertex=" 9 2 0
-10 10 10
9 -2 0
10 3 -10
10 -3 -10
-8 10 -10
-10 -10 10
-8 -10 -10
-5 0 20"/>
</asset>
</mujoco>)");

auto namefile = createTempFile(xmlDataNoStrip);

typedef ::pinocchio::mjcf::details::MjcfGraph MjcfGraph;
pinocchio::Model model_m;
MjcfGraph::UrdfVisitor visitor(model_m);

MjcfGraph graph(visitor, "/fakeMjcf/fake.xml");
graph.parseGraphFromXML(namefile.name());

// Test Meshes
pinocchio::mjcf::details::MjcfMesh mesh = graph.mapOfMeshes.at("chasis");
BOOST_CHECK_EQUAL(mesh.scale, Eigen::Vector3d(0.01, 0.006, 0.0015));
hpp::fcl::MatrixX3s vertices(3, 9);
vertices << 9, 2, 0, -10, 10, 10, 9, -2, 0, 10, 3, -10, 10, -3, -10, -8, 10, -10, -10, -10, 10,
-8, -10, -10, -5, 0, 20;
BOOST_CHECK_EQUAL(mesh.vertices.rows(), 9);
for (auto i = 0; i < mesh.vertices.rows(); ++i)
{
BOOST_CHECK(mesh.vertices.row(i) == vertices.row(i));
}
}

BOOST_AUTO_TEST_SUITE_END()

0 comments on commit f0d5a0e

Please sign in to comment.