#ifndef OPENMESH_PYTHON_MESH_HH #define OPENMESH_PYTHON_MESH_HH #include "Utilities.hh" #include "MeshTypes.hh" #include "Iterator.hh" #include "Circulator.hh" #include #include #include namespace py = pybind11; namespace OM = OpenMesh; /** * Return value policy for functions that return references to objects that are * managed by %OpenMesh. */ #define OPENMESH_PYTHON_DEFAULT_POLICY py::return_value_policy::copy /** * Set the status of an item. * * @tparam Mesh A mesh type. * @tparam PropHandle A handle type. * * @param _self The mesh instance that is to be used. * @param _h The handle of the item whose status is to be set. * @param _info The status to be set. * * Depending on @ref OPENMESH_PYTHON_DEFAULT_POLICY, Mesh::status may * return by value instead of reference. This function ensures that the * status of an item can be changed nonetheless. */ template void set_status(Mesh& _self, IndexHandle _h, const OpenMesh::Attributes::StatusInfo& _info) { _self.status(_h) = _info; } /** * Thin wrapper for assign_connectivity. * * @tparam Mesh A mesh type. * @tparam OtherMesh A mesh type. * * @param _self The mesh instance that is to be used. * @param _other The mesh from which the connectivity is to be copied. */ template void assign_connectivity(Mesh& _self, const OtherMesh& _other) { _self.assign_connectivity(_other); } /** * Get an iterator. */ template IteratorWrapperT get_iterator(Mesh& _self) { return IteratorWrapperT(_self, typename Iterator::value_type(0)); } /** * Get a skipping iterator. */ template IteratorWrapperT get_skipping_iterator(Mesh& _self) { return IteratorWrapperT(_self, typename Iterator::value_type(0), true); } /** * Get a circulator. * * @tparam Mesh A Mesh type. * @tparam Circulator A circulator type. * @tparam CenterEntityHandle The appropriate handle type. * * @param _self The mesh instance that is to be used. * @param _handle The handle of the item to circulate around. */ template CirculatorWrapperT get_circulator(Mesh& _self, CenterEntityHandle _handle) { return CirculatorWrapperT(_self, _handle); } /** * Garbage collection using lists instead of vectors to keep track of a set of * handles. * * @tparam Mesh A Mesh type. * * @param _self The mesh instance that is to be used. * @param _vh_to_update The list of vertex handles to be updated. * @param _hh_to_update The list of halfedge handles to be updated. * @param _fh_to_update The list of face handles to be updated. * @param _v Remove deleted vertices? * @param _e Remove deleted edges? * @param _f Remove deleted faces? */ template void garbage_collection(Mesh& _self, py::list& _vh_to_update, py::list& _hh_to_update, py::list& _fh_to_update, bool _v = true, bool _e = true, bool _f = true) { // Convert list of handles to vector of pointers std::vector vh_vector; for (auto item : _vh_to_update) { if (py::isinstance(item)) { vh_vector.push_back(item.cast()); } } // Convert list of handles to vector of pointers std::vector hh_vector; for (auto item : _hh_to_update) { if (py::isinstance(item)) { hh_vector.push_back(item.cast()); } } // Convert list of handles to vector of pointers std::vector fh_vector; for (auto item : _fh_to_update) { if (py::isinstance(item)) { fh_vector.push_back(item.cast()); } } // Call garbage collection _self.garbage_collection(vh_vector, hh_vector, fh_vector, _v, _e, _f); } /** * Add a new face from a %Python list of vertex handles. * * @tparam Mesh A Mesh type. * * @param _self The mesh instance that is to be used. * @param _vhandles The list of vertex handles. */ template OM::FaceHandle add_face(Mesh& _self, const py::list& _vhandles) { // TODO Use automatic conversion instead? std::vector vector; for (auto item : _vhandles) { if (py::isinstance(item)) { vector.push_back(item.cast()); } } return _self.add_face(vector); } /** * Converts OpenMesh vectors to numpy arrays. * * @tparam vector A Vector type. * @param _vec The vector to be converted. */ template py::array_t vec2numpy(const Vector& _vec) { typedef typename Vector::value_type dtype; dtype *data = new dtype[_vec.size()]; std::copy_n(_vec.data(), _vec.size(), data); py::capsule base = free_when_done(data); return py::array_t({_vec.size()}, {sizeof(dtype)}, data, base); } /** * Converts OpenMesh vectors to numpy arrays. * * The returned array references the vector's underlying data, i.e. changes * made to the returned array affect the original mesh. * * @tparam Mesh A Mesh type. * @tparam vector A Vector type. * * @param _mesh The mesh that owns the vector's underlying memory. In order * to avaoid dangling pointers, the lifetime of this mesh is tied to the * lifetime of the returned numpy array. * @param _vec The vector to be converted. */ template py::array_t vec2numpy(Mesh& _mesh, Vector& _vec, size_t _n = 1) { typedef typename Vector::value_type dtype; std::vector shape; std::vector strides; if (_n == 1) { shape = {_vec.size()}; strides = {sizeof(dtype)}; } else { shape = {_n, _vec.size()}; strides = {_vec.size() * sizeof(dtype), sizeof(dtype)}; } return py::array_t(shape, strides, _vec.data(), py::cast(_mesh)); } template py::array_t flt2numpy(Mesh& _mesh, const float& _flt, size_t _n = 1) { return py::array_t({_n}, {sizeof(float)}, &_flt, py::cast(_mesh)); } template py::array_t flt2numpy(Mesh& _mesh, const double& _flt, size_t _n = 1) { return py::array_t({_n}, {sizeof(double)}, &_flt, py::cast(_mesh)); } /** * Returns the face indices of this mesh as a numpy array. * * Note that the array is constructed on the fly and does not * reference the underlying mesh. */ py::array_t face_vertex_indices_trimesh(TriMesh& _self) { if (_self.n_faces() == 0) { return py::array_t(); } int *indices = new int[_self.n_faces() * 3]; for (auto fh : _self.faces()) { auto fv_it = _self.fv_iter(fh); indices[fh.idx() * 3 + 0] = fv_it->idx(); ++fv_it; indices[fh.idx() * 3 + 1] = fv_it->idx(); ++fv_it; indices[fh.idx() * 3 + 2] = fv_it->idx(); } const auto shape = {_self.n_faces(), size_t(3)}; const auto strides = {3 * sizeof(int), sizeof(int)}; py::capsule base = free_when_done(indices); return py::array_t(shape, strides, indices, base); } /** * Returns the face indices of this mesh as a numpy array. * * If the faces of the mesh have different valences, the array * is padded with -1 entries. * * Note that the array is constructed on the fly and does not * reference the underlying mesh. */ py::array_t face_vertex_indices_polymesh(PolyMesh& _self) { if (_self.n_faces() == 0) { return py::array_t(); } int max_valence = 0; for (auto fh : _self.faces()) { max_valence = std::max(max_valence, int(_self.valence(fh))); } int *indices = new int[_self.n_faces() * max_valence]; for (auto fh : _self.faces()) { auto fv_it = _self.fv_iter(fh); const int valence = _self.valence(fh); for (int i = 0; i < valence; ++i) { indices[fh.idx() * max_valence + i] = fv_it->idx(); ++fv_it; } for (int i = valence; i < max_valence; ++i) { indices[fh.idx() * max_valence + i] = -1; } } const auto shape = {_self.n_faces(), size_t(max_valence)}; const auto strides = {max_valence * sizeof(int), sizeof(int)}; py::capsule base = free_when_done(indices); return py::array_t(shape, strides, indices, base); } /** * Returns the edge indices of this mesh as a numpy array. * * Note that the array is constructed on the fly and does not * reference the underlying mesh. */ template py::array_t edge_vertex_indices(Mesh& _self) { if (_self.n_edges() == 0) { return py::array_t(); } int *indices = new int[_self.n_edges() * 2]; for (auto eh : _self.edges()) { auto heh = _self.halfedge_handle(eh, 0); auto vh1 = _self.from_vertex_handle(heh); auto vh2 = _self.to_vertex_handle(heh); indices[eh.idx() * 2 + 0] = vh1.idx(); indices[eh.idx() * 2 + 1] = vh2.idx(); } const auto shape = {_self.n_edges(), size_t(2)}; const auto strides = {2 * sizeof(int), sizeof(int)}; py::capsule base = free_when_done(indices); return py::array_t(shape, strides, indices, base); } /** * Returns the halfedge indices of this mesh as a numpy array. * * Note that the array is constructed on the fly and does not * reference the underlying mesh. */ template py::array_t halfedge_vertex_indices(Mesh& _self) { if (_self.n_halfedges() == 0) { return py::array_t(); } int *indices = new int[_self.n_halfedges() * 2]; for (auto heh : _self.halfedges()) { auto vh1 = _self.from_vertex_handle(heh); auto vh2 = _self.to_vertex_handle(heh); indices[heh.idx() * 2 + 0] = vh1.idx(); indices[heh.idx() * 2 + 1] = vh2.idx(); } const auto shape = {_self.n_halfedges(), size_t(2)}; const auto strides = {2 * sizeof(int), sizeof(int)}; py::capsule base = free_when_done(indices); return py::array_t(shape, strides, indices, base); } /** * This function template is used to expose mesh member functions that are only * available for a specific type of mesh (i.e. they are available for polygon * meshes or triangle meshes, but not both). * * @tparam Class A pybind11::class type. * * @param _class The pybind11::class instance for which the member * functions are to be defined. */ template void expose_type_specific_functions(Class& _class) { // See the template specializations below } /** * Function template specialization for polygon meshes. */ template <> void expose_type_specific_functions(py::class_& _class) { typedef PolyMesh::Scalar Scalar; typedef PolyMesh::Point Point; typedef PolyMesh::Normal Normal; typedef PolyMesh::Color Color; typedef py::array_t np_point_t; OM::FaceHandle (PolyMesh::*add_face_3_vh)(OM::VertexHandle, OM::VertexHandle, OM::VertexHandle ) = &PolyMesh::add_face; OM::FaceHandle (PolyMesh::*add_face_4_vh)(OM::VertexHandle, OM::VertexHandle, OM::VertexHandle, OM::VertexHandle) = &PolyMesh::add_face; OM::FaceHandle (*add_face_list)(PolyMesh&, const py::list&) = &add_face; _class .def("add_face", add_face_3_vh) .def("add_face", add_face_4_vh) .def("add_face", add_face_list) .def("split", [](PolyMesh& _self, OM::EdgeHandle _eh, np_point_t _arr) { _self.split(_eh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("split", [](PolyMesh& _self, OM::FaceHandle _fh, np_point_t _arr) { _self.split(_fh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("insert_edge", &PolyMesh::insert_edge) .def("face_vertex_indices", &face_vertex_indices_polymesh) .def("fv_indices", &face_vertex_indices_polymesh) .def("calc_face_normal", [](PolyMesh& _self, np_point_t _p0, np_point_t _p1, np_point_t _p2) { const Point p0(_p0.at(0), _p0.at(1), _p0.at(2)); const Point p1(_p1.at(0), _p1.at(1), _p1.at(2)); const Point p2(_p2.at(0), _p2.at(1), _p2.at(2)); return vec2numpy(_self.calc_face_normal(p0, p1, p2)); }) ; } /** * Function template specialization for triangle meshes. */ template <> void expose_type_specific_functions(py::class_& _class) { typedef TriMesh::Scalar Scalar; typedef TriMesh::Point Point; typedef TriMesh::Normal Normal; typedef TriMesh::Color Color; typedef py::array_t np_point_t; OM::FaceHandle (TriMesh::*add_face_3_vh)(OM::VertexHandle, OM::VertexHandle, OM::VertexHandle) = &TriMesh::add_face; OM::FaceHandle (*add_face_list)(TriMesh&, const py::list&) = &add_face; void (TriMesh::*split_copy_eh_vh)(OM::EdgeHandle, OM::VertexHandle) = &TriMesh::split_copy; OM::HalfedgeHandle (TriMesh::*vertex_split_vh)(OM::VertexHandle, OM::VertexHandle, OM::VertexHandle, OM::VertexHandle) = &TriMesh::vertex_split; _class .def("add_face", add_face_3_vh) .def("add_face", add_face_list) .def("split", [](TriMesh& _self, OM::EdgeHandle _eh, np_point_t _arr) { return _self.split(_eh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("split", [](TriMesh& _self, OM::FaceHandle _fh, np_point_t _arr) { return _self.split(_fh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("split_copy", split_copy_eh_vh) .def("split_copy", [](TriMesh& _self, OM::EdgeHandle _eh, np_point_t _arr) { return _self.split_copy(_eh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("split_copy", [](TriMesh& _self, OM::FaceHandle _fh, np_point_t _arr) { return _self.split_copy(_fh, Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("opposite_vh", &TriMesh::opposite_vh) .def("opposite_he_opposite_vh", &TriMesh::opposite_he_opposite_vh) .def("vertex_split", vertex_split_vh) .def("vertex_split", [](TriMesh& _self, np_point_t _arr, OM::VertexHandle _v1, OM::VertexHandle _vl, OM::VertexHandle _vr) { return _self.vertex_split(Point(_arr.at(0), _arr.at(1), _arr.at(2)), _v1, _vl, _vr); }) .def("is_flip_ok", &TriMesh::is_flip_ok) .def("flip", &TriMesh::flip) .def("face_vertex_indices", &face_vertex_indices_trimesh) .def("fv_indices", &face_vertex_indices_trimesh) ; } /** * Expose a mesh type to %Python. * * @tparam Mesh A mesh type. * * @param _name The name of the mesh type to be exposed. */ template void expose_mesh(py::module& m, const char *_name) { using OpenMesh::Attributes::StatusInfo; typedef typename Mesh::Scalar Scalar; typedef typename Mesh::Point Point; typedef typename Mesh::Normal Normal; typedef typename Mesh::Color Color; typedef typename Mesh::TexCoord1D TexCoord1D; typedef typename Mesh::TexCoord2D TexCoord2D; typedef typename Mesh::TexCoord3D TexCoord3D; //====================================================================== // KernelT Function Pointers //====================================================================== // Get the i'th item OM::VertexHandle (Mesh::*vertex_handle_uint )(unsigned int) const = &Mesh::vertex_handle; OM::HalfedgeHandle (Mesh::*halfedge_handle_uint)(unsigned int) const = &Mesh::halfedge_handle; OM::EdgeHandle (Mesh::*edge_handle_uint )(unsigned int) const = &Mesh::edge_handle; OM::FaceHandle (Mesh::*face_handle_uint )(unsigned int) const = &Mesh::face_handle; // Delete items void (Mesh::*garbage_collection_bools)(bool, bool, bool) = &Mesh::garbage_collection; void (*garbage_collection_lists_bools)(Mesh&, py::list&, py::list&, py::list&, bool, bool, bool) = &garbage_collection; // Vertex connectivity OM::HalfedgeHandle (Mesh::*halfedge_handle_vh)(OM::VertexHandle) const = &Mesh::halfedge_handle; OM::HalfedgeHandle (Mesh::*halfedge_handle_fh)(OM::FaceHandle ) const = &Mesh::halfedge_handle; // Halfedge connectivity OM::FaceHandle (Mesh::*face_handle_hh )(OM::HalfedgeHandle) const = &Mesh::face_handle; OM::HalfedgeHandle (Mesh::*prev_halfedge_handle_hh)(OM::HalfedgeHandle) const = &Mesh::prev_halfedge_handle; OM::EdgeHandle (Mesh::*edge_handle_hh )(OM::HalfedgeHandle) const = &Mesh::edge_handle; // Edge connectivity OM::HalfedgeHandle (Mesh::*halfedge_handle_eh_uint)(OM::EdgeHandle, unsigned int) const = &Mesh::halfedge_handle; // Set halfedge void (Mesh::*set_halfedge_handle_vh_hh)(OM::VertexHandle, OM::HalfedgeHandle) = &Mesh::set_halfedge_handle; void (Mesh::*set_halfedge_handle_fh_hh)(OM::FaceHandle, OM::HalfedgeHandle) = &Mesh::set_halfedge_handle; // Handle -> Item const typename Mesh::Vertex& (Mesh::*vertex )(OM::VertexHandle ) const = &Mesh::vertex; const typename Mesh::Halfedge& (Mesh::*halfedge)(OM::HalfedgeHandle) const = &Mesh::halfedge; const typename Mesh::Edge& (Mesh::*edge )(OM::EdgeHandle ) const = &Mesh::edge; const typename Mesh::Face& (Mesh::*face )(OM::FaceHandle ) const = &Mesh::face; // Item -> Handle OM::VertexHandle (Mesh::*handle_v)(const typename Mesh::Vertex& ) const = &Mesh::handle; OM::HalfedgeHandle (Mesh::*handle_h)(const typename Mesh::Halfedge&) const = &Mesh::handle; OM::EdgeHandle (Mesh::*handle_e)(const typename Mesh::Edge& ) const = &Mesh::handle; OM::FaceHandle (Mesh::*handle_f)(const typename Mesh::Face& ) const = &Mesh::handle; // Get value of a standard property (status) const StatusInfo& (Mesh::*status_vh)(OM::VertexHandle ) const = &Mesh::status; const StatusInfo& (Mesh::*status_hh)(OM::HalfedgeHandle) const = &Mesh::status; const StatusInfo& (Mesh::*status_eh)(OM::EdgeHandle ) const = &Mesh::status; const StatusInfo& (Mesh::*status_fh)(OM::FaceHandle ) const = &Mesh::status; // Set value of a standard property (status) void (*set_status_vh)(Mesh&, OM::VertexHandle, const StatusInfo&) = &set_status; void (*set_status_hh)(Mesh&, OM::HalfedgeHandle, const StatusInfo&) = &set_status; void (*set_status_eh)(Mesh&, OM::EdgeHandle, const StatusInfo&) = &set_status; void (*set_status_fh)(Mesh&, OM::FaceHandle, const StatusInfo&) = &set_status; // Low-level adding new items OM::VertexHandle (Mesh::*new_vertex_void )(void ) = &Mesh::new_vertex; OM::FaceHandle (Mesh::*new_face_void )(void ) = &Mesh::new_face; OM::FaceHandle (Mesh::*new_face_face )(const typename Mesh::Face& ) = &Mesh::new_face; // Kernel item iterators IteratorWrapperT (*vertices )(Mesh&) = &get_iterator; IteratorWrapperT (*halfedges)(Mesh&) = &get_iterator; IteratorWrapperT (*edges )(Mesh&) = &get_iterator; IteratorWrapperT (*faces )(Mesh&) = &get_iterator; IteratorWrapperT (*svertices )(Mesh&) = &get_skipping_iterator; IteratorWrapperT (*shalfedges)(Mesh&) = &get_skipping_iterator; IteratorWrapperT (*sedges )(Mesh&) = &get_skipping_iterator; IteratorWrapperT (*sfaces )(Mesh&) = &get_skipping_iterator; //====================================================================== // BaseKernel Function Pointers //====================================================================== // Copy all properties void (Mesh::*copy_all_properties_vh_vh_bool)(OM::VertexHandle, OM::VertexHandle, bool) = &Mesh::copy_all_properties; void (Mesh::*copy_all_properties_hh_hh_bool)(OM::HalfedgeHandle, OM::HalfedgeHandle, bool) = &Mesh::copy_all_properties; void (Mesh::*copy_all_properties_eh_eh_bool)(OM::EdgeHandle, OM::EdgeHandle, bool) = &Mesh::copy_all_properties; void (Mesh::*copy_all_properties_fh_fh_bool)(OM::FaceHandle, OM::FaceHandle, bool) = &Mesh::copy_all_properties; //====================================================================== // PolyConnectivity Function Pointers //====================================================================== // Assign connectivity void (*assign_connectivity_poly)(Mesh&, const PolyMesh&) = &assign_connectivity; void (*assign_connectivity_tri )(Mesh&, const TriMesh& ) = &assign_connectivity; // Vertex and face valence unsigned int (Mesh::*valence_vh)(OM::VertexHandle) const = &Mesh::valence; unsigned int (Mesh::*valence_fh)(OM::FaceHandle ) const = &Mesh::valence; // Triangulate face or mesh void (Mesh::*triangulate_fh )(OM::FaceHandle) = &Mesh::triangulate; void (Mesh::*triangulate_void)( ) = &Mesh::triangulate; // Vertex and Face circulators CirculatorWrapperT (*vv )(Mesh&, OM::VertexHandle ) = &get_circulator; CirculatorWrapperT (*vih)(Mesh&, OM::VertexHandle ) = &get_circulator; CirculatorWrapperT (*voh)(Mesh&, OM::VertexHandle ) = &get_circulator; CirculatorWrapperT (*ve )(Mesh&, OM::VertexHandle ) = &get_circulator; CirculatorWrapperT (*vf )(Mesh&, OM::VertexHandle ) = &get_circulator; CirculatorWrapperT (*fv )(Mesh&, OM::FaceHandle ) = &get_circulator; CirculatorWrapperT (*fh )(Mesh&, OM::FaceHandle ) = &get_circulator; CirculatorWrapperT (*fe )(Mesh&, OM::FaceHandle ) = &get_circulator; CirculatorWrapperT (*ff )(Mesh&, OM::FaceHandle ) = &get_circulator; CirculatorWrapperT (*hl )(Mesh&, OM::HalfedgeHandle) = &get_circulator; // Boundary and manifold tests bool (Mesh::*is_boundary_hh)(OM::HalfedgeHandle ) const = &Mesh::is_boundary; bool (Mesh::*is_boundary_eh)(OM::EdgeHandle ) const = &Mesh::is_boundary; bool (Mesh::*is_boundary_vh)(OM::VertexHandle ) const = &Mesh::is_boundary; bool (Mesh::*is_boundary_fh)(OM::FaceHandle, bool) const = &Mesh::is_boundary; // Generic handle derefertiation const typename Mesh::Vertex& (Mesh::*deref_vh)(OM::VertexHandle ) const = &Mesh::deref; const typename Mesh::Halfedge& (Mesh::*deref_hh)(OM::HalfedgeHandle) const = &Mesh::deref; const typename Mesh::Edge& (Mesh::*deref_eh)(OM::EdgeHandle ) const = &Mesh::deref; const typename Mesh::Face& (Mesh::*deref_fh)(OM::FaceHandle ) const = &Mesh::deref; //====================================================================== // PolyMeshT Function Pointers //====================================================================== Scalar (Mesh::*calc_edge_length_eh)(OM::EdgeHandle ) const = &Mesh::calc_edge_length; Scalar (Mesh::*calc_edge_length_hh)(OM::HalfedgeHandle) const = &Mesh::calc_edge_length; Scalar (Mesh::*calc_edge_sqr_length_eh)(OM::EdgeHandle ) const = &Mesh::calc_edge_sqr_length; Scalar (Mesh::*calc_edge_sqr_length_hh)(OM::HalfedgeHandle) const = &Mesh::calc_edge_sqr_length; Scalar (Mesh::*calc_dihedral_angle_fast_hh)(OM::HalfedgeHandle) const = &Mesh::calc_dihedral_angle_fast; Scalar (Mesh::*calc_dihedral_angle_fast_eh)(OM::EdgeHandle ) const = &Mesh::calc_dihedral_angle_fast; Scalar (Mesh::*calc_dihedral_angle_hh)(OM::HalfedgeHandle) const = &Mesh::calc_dihedral_angle; Scalar (Mesh::*calc_dihedral_angle_eh)(OM::EdgeHandle ) const = &Mesh::calc_dihedral_angle; unsigned int (Mesh::*find_feature_edges)(Scalar) = &Mesh::find_feature_edges; void (Mesh::*split_fh_vh)(OM::FaceHandle, OM::VertexHandle) = &Mesh::split; void (Mesh::*split_eh_vh)(OM::EdgeHandle, OM::VertexHandle) = &Mesh::split; void (Mesh::*split_copy_fh_vh)(OM::FaceHandle, OM::VertexHandle) = &Mesh::split_copy; //====================================================================== // Mesh Type //====================================================================== py::class_ class_mesh(m, _name); class_mesh .def(py::init<>()) //====================================================================== // KernelT //====================================================================== .def("reserve", &Mesh::reserve) .def("vertex", vertex, py::return_value_policy::reference) .def("halfedge", halfedge, py::return_value_policy::reference) .def("edge", edge, py::return_value_policy::reference) .def("face", face, py::return_value_policy::reference) .def("handle", handle_v) .def("handle", handle_h) .def("handle", handle_e) .def("handle", handle_f) .def("vertex_handle", vertex_handle_uint) .def("halfedge_handle", halfedge_handle_uint) .def("edge_handle", edge_handle_uint) .def("face_handle", face_handle_uint) .def("clear", &Mesh::clear) .def("clean", &Mesh::clean) .def("garbage_collection", garbage_collection_bools, py::arg("v")=true, py::arg("e")=true, py::arg("f")=true) .def("garbage_collection", garbage_collection_lists_bools, py::arg("vh_to_update"), py::arg("hh_to_update"), py::arg("fh_to_update"), py::arg("v")=true, py::arg("e")=true, py::arg("f")=true) .def("n_vertices", &Mesh::n_vertices) .def("n_halfedges", &Mesh::n_halfedges) .def("n_edges", &Mesh::n_edges) .def("n_faces", &Mesh::n_faces) .def("vertices_empty", &Mesh::vertices_empty) .def("halfedges_empty", &Mesh::halfedges_empty) .def("edges_empty", &Mesh::edges_empty) .def("faces_empty", &Mesh::faces_empty) .def("halfedge_handle", halfedge_handle_vh) .def("set_halfedge_handle", set_halfedge_handle_vh_hh) .def("to_vertex_handle", &Mesh::to_vertex_handle) .def("from_vertex_handle", &Mesh::from_vertex_handle) .def("set_vertex_handle", &Mesh::set_vertex_handle) .def("face_handle", face_handle_hh) .def("set_face_handle", &Mesh::set_face_handle) .def("next_halfedge_handle", &Mesh::next_halfedge_handle) .def("set_next_halfedge_handle", &Mesh::set_next_halfedge_handle) .def("prev_halfedge_handle", prev_halfedge_handle_hh) .def("opposite_halfedge_handle", &Mesh::opposite_halfedge_handle) .def("ccw_rotated_halfedge_handle", &Mesh::ccw_rotated_halfedge_handle) .def("cw_rotated_halfedge_handle", &Mesh::cw_rotated_halfedge_handle) .def("edge_handle", edge_handle_hh) .def("halfedge_handle", halfedge_handle_eh_uint) .def("halfedge_handle", halfedge_handle_fh) .def("set_halfedge_handle", set_halfedge_handle_fh_hh) .def("status", status_vh, OPENMESH_PYTHON_DEFAULT_POLICY) .def("set_status", set_status_vh) .def("status", status_hh, OPENMESH_PYTHON_DEFAULT_POLICY) .def("set_status", set_status_hh) .def("status", status_eh, OPENMESH_PYTHON_DEFAULT_POLICY) .def("set_status", set_status_eh) .def("status", status_fh, OPENMESH_PYTHON_DEFAULT_POLICY) .def("set_status", set_status_fh) .def("request_vertex_normals", &Mesh::request_vertex_normals) .def("request_vertex_colors", &Mesh::request_vertex_colors) .def("request_vertex_texcoords1D", &Mesh::request_vertex_texcoords1D) .def("request_vertex_texcoords2D", &Mesh::request_vertex_texcoords2D) .def("request_vertex_texcoords3D", &Mesh::request_vertex_texcoords3D) .def("request_vertex_status", &Mesh::request_vertex_status) .def("request_halfedge_status", &Mesh::request_halfedge_status) .def("request_halfedge_normals", &Mesh::request_halfedge_normals) .def("request_halfedge_colors", &Mesh::request_halfedge_colors) .def("request_halfedge_texcoords1D", &Mesh::request_halfedge_texcoords1D) .def("request_halfedge_texcoords2D", &Mesh::request_halfedge_texcoords2D) .def("request_halfedge_texcoords3D", &Mesh::request_halfedge_texcoords3D) .def("request_edge_status", &Mesh::request_edge_status) .def("request_edge_colors", &Mesh::request_edge_colors) .def("request_face_normals", &Mesh::request_face_normals) .def("request_face_colors", &Mesh::request_face_colors) .def("request_face_status", &Mesh::request_face_status) .def("request_face_texture_index", &Mesh::request_face_texture_index) .def("release_vertex_normals", &Mesh::release_vertex_normals) .def("release_vertex_colors", &Mesh::release_vertex_colors) .def("release_vertex_texcoords1D", &Mesh::release_vertex_texcoords1D) .def("release_vertex_texcoords2D", &Mesh::release_vertex_texcoords2D) .def("release_vertex_texcoords3D", &Mesh::release_vertex_texcoords3D) .def("release_vertex_status", &Mesh::release_vertex_status) .def("release_halfedge_status", &Mesh::release_halfedge_status) .def("release_halfedge_normals", &Mesh::release_halfedge_normals) .def("release_halfedge_colors", &Mesh::release_halfedge_colors) .def("release_halfedge_texcoords1D", &Mesh::release_halfedge_texcoords1D) .def("release_halfedge_texcoords2D", &Mesh::release_halfedge_texcoords2D) .def("release_halfedge_texcoords3D", &Mesh::release_halfedge_texcoords3D) .def("release_edge_status", &Mesh::release_edge_status) .def("release_edge_colors", &Mesh::release_edge_colors) .def("release_face_normals", &Mesh::release_face_normals) .def("release_face_colors", &Mesh::release_face_colors) .def("release_face_status", &Mesh::release_face_status) .def("release_face_texture_index", &Mesh::release_face_texture_index) .def("has_vertex_normals", &Mesh::has_vertex_normals) .def("has_vertex_colors", &Mesh::has_vertex_colors) .def("has_vertex_texcoords1D", &Mesh::has_vertex_texcoords1D) .def("has_vertex_texcoords2D", &Mesh::has_vertex_texcoords2D) .def("has_vertex_texcoords3D", &Mesh::has_vertex_texcoords3D) .def("has_vertex_status", &Mesh::has_vertex_status) .def("has_halfedge_status", &Mesh::has_halfedge_status) .def("has_halfedge_normals", &Mesh::has_halfedge_normals) .def("has_halfedge_colors", &Mesh::has_halfedge_colors) .def("has_halfedge_texcoords1D", &Mesh::has_halfedge_texcoords1D) .def("has_halfedge_texcoords2D", &Mesh::has_halfedge_texcoords2D) .def("has_halfedge_texcoords3D", &Mesh::has_halfedge_texcoords3D) .def("has_edge_status", &Mesh::has_edge_status) .def("has_edge_colors", &Mesh::has_edge_colors) .def("has_face_normals", &Mesh::has_face_normals) .def("has_face_colors", &Mesh::has_face_colors) .def("has_face_status", &Mesh::has_face_status) .def("has_face_texture_index", &Mesh::has_face_texture_index) .def("new_vertex", new_vertex_void) .def("new_vertex", [](Mesh& _self, py::array_t _arr) { return _self.new_vertex(Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("new_edge", &Mesh::new_edge) .def("new_face", new_face_void) .def("new_face", new_face_face) .def("vertices", vertices) .def("halfedges", halfedges) .def("edges", edges) .def("faces", faces) .def("svertices", svertices) .def("shalfedges", shalfedges) .def("sedges", sedges) .def("sfaces", sfaces) //====================================================================== // BaseKernel //====================================================================== .def("copy_all_properties", copy_all_properties_vh_vh_bool, py::arg("vh_from"), py::arg("vh_to"), py::arg("copy_build_in")=false) .def("copy_all_properties", copy_all_properties_hh_hh_bool, py::arg("hh_from"), py::arg("hh_to"), py::arg("copy_build_in")=false) .def("copy_all_properties", copy_all_properties_eh_eh_bool, py::arg("eh_from"), py::arg("eh_to"), py::arg("copy_build_in")=false) .def("copy_all_properties", copy_all_properties_fh_fh_bool, py::arg("fh_from"), py::arg("fh_to"), py::arg("copy_build_in")=false) //====================================================================== // PolyConnectivity //====================================================================== .def("assign_connectivity", assign_connectivity_poly) .def("assign_connectivity", assign_connectivity_tri) .def("opposite_face_handle", &Mesh::opposite_face_handle) .def("adjust_outgoing_halfedge", &Mesh::adjust_outgoing_halfedge) .def("find_halfedge", &Mesh::find_halfedge) .def("valence", valence_vh) .def("valence", valence_fh) .def("collapse", &Mesh::collapse) .def("is_simple_link", &Mesh::is_simple_link) .def("is_simply_connected", &Mesh::is_simply_connected) .def("remove_edge", &Mesh::remove_edge) .def("reinsert_edge", &Mesh::reinsert_edge) .def("triangulate", triangulate_fh) .def("triangulate", triangulate_void) .def("split_edge", &Mesh::split_edge) .def("split_edge_copy", &Mesh::split_edge_copy) .def("add_vertex", [](Mesh& _self, py::array_t _arr) { return _self.add_vertex(Point(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("is_collapse_ok", &Mesh::is_collapse_ok) .def("delete_vertex", [](Mesh& _self, OM::VertexHandle _vh, bool _delete_isolated) { if (!_self.has_vertex_status()) _self.request_vertex_status(); if (!_self.has_face_status()) _self.request_face_status(); _self.delete_vertex(_vh, _delete_isolated); }, py::arg("vh"), py::arg("delete_isolated_vertices")=true) .def("delete_edge", [](Mesh& _self, OM::EdgeHandle _eh, bool _delete_isolated) { if (!_self.has_vertex_status() && _delete_isolated) _self.request_vertex_status(); if (!_self.has_edge_status()) _self.request_edge_status(); if (!_self.has_face_status()) _self.request_face_status(); _self.delete_edge(_eh, _delete_isolated); }, py::arg("eh"), py::arg("delete_isolated_vertices")=true) .def("delete_face", [](Mesh& _self, OM::FaceHandle _fh, bool _delete_isolated) { if (!_self.has_vertex_status() && _delete_isolated) _self.request_vertex_status(); if (!_self.has_face_status()) _self.request_face_status(); _self.delete_face(_fh, _delete_isolated); }, py::arg("fh"), py::arg("delete_isolated_vertices")=true) .def("vv", vv) .def("vih", vih) .def("voh", voh) .def("ve", ve) .def("vf", vf) .def("fv", fv) .def("fh", fh) .def("fe", fe) .def("ff", ff) .def("hl", hl) .def("is_boundary", is_boundary_hh) .def("is_boundary", is_boundary_eh) .def("is_boundary", is_boundary_vh) .def("is_boundary", is_boundary_fh, py::arg("fh"), py::arg("check_vertex")=false) .def("is_manifold", &Mesh::is_manifold) .def("deref", deref_vh, py::return_value_policy::reference) .def("deref", deref_hh, py::return_value_policy::reference) .def("deref", deref_eh, py::return_value_policy::reference) .def("deref", deref_fh, py::return_value_policy::reference) .def_static("is_triangles", &Mesh::is_triangles) .def_readonly_static("InvalidVertexHandle", &Mesh::InvalidVertexHandle) .def_readonly_static("InvalidHalfedgeHandle", &Mesh::InvalidHalfedgeHandle) .def_readonly_static("InvalidEdgeHandle", &Mesh::InvalidEdgeHandle) .def_readonly_static("InvalidFaceHandle", &Mesh::InvalidFaceHandle) //====================================================================== // PolyMeshT //====================================================================== .def("calc_edge_length", calc_edge_length_eh) .def("calc_edge_length", calc_edge_length_hh) .def("calc_edge_sqr_length", calc_edge_sqr_length_eh) .def("calc_edge_sqr_length", calc_edge_sqr_length_hh) .def("calc_sector_angle", &Mesh::calc_sector_angle) .def("calc_sector_area", &Mesh::calc_sector_area) .def("calc_dihedral_angle_fast", calc_dihedral_angle_fast_hh) .def("calc_dihedral_angle_fast", calc_dihedral_angle_fast_eh) .def("calc_dihedral_angle", calc_dihedral_angle_hh) .def("calc_dihedral_angle", calc_dihedral_angle_eh) .def("find_feature_edges", find_feature_edges, py::arg("angle_tresh")=OM::deg_to_rad(44.0)) .def("split", split_fh_vh) .def("split", split_eh_vh) .def("split_copy", split_copy_fh_vh) .def("update_normals", [](Mesh& _self) { if (!_self.has_face_normals()) { _self.request_face_normals(); } if (!_self.has_halfedge_normals()) { _self.request_halfedge_normals(); } if (!_self.has_vertex_normals()) { _self.request_vertex_normals(); } _self.update_normals(); }) .def("update_normal", [](Mesh& _self, OM::FaceHandle _fh) { if (!_self.has_face_normals()) _self.request_face_normals(); _self.update_normal(_fh); }) .def("update_face_normals", [](Mesh& _self) { if (!_self.has_face_normals()) _self.request_face_normals(); _self.update_face_normals(); }) .def("update_normal", [](Mesh& _self, OM::HalfedgeHandle _hh, double _feature_angle) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } if (!_self.has_halfedge_normals()) { _self.request_halfedge_normals(); } _self.update_normal(_hh, _feature_angle); }, py::arg("heh"), py::arg("feature_angle")=0.8) .def("update_halfedge_normals", [](Mesh& _self, double _feature_angle) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } if (!_self.has_halfedge_normals()) { _self.request_halfedge_normals(); } _self.update_halfedge_normals(_feature_angle); }, py::arg("feature_angle")=0.8) .def("update_normal", [](Mesh& _self, OM::VertexHandle _vh) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } if (!_self.has_vertex_normals()) { _self.request_vertex_normals(); } _self.update_normal(_vh); }) .def("update_vertex_normals", [](Mesh& _self) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } if (!_self.has_vertex_normals()) { _self.request_vertex_normals(); } _self.update_vertex_normals(); }) .def("is_estimated_feature_edge", &Mesh::is_estimated_feature_edge) .def_static("is_polymesh", &Mesh::is_polymesh) .def("is_trimesh", &Mesh::is_trimesh) //====================================================================== // numpy calc_* //====================================================================== .def("calc_face_normal", [](Mesh& _self, OM::FaceHandle _fh) { return vec2numpy(_self.calc_face_normal(_fh)); }) .def("calc_halfedge_normal", [](Mesh& _self, OM::HalfedgeHandle _heh, double _feature_angle) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } return vec2numpy(_self.calc_halfedge_normal(_heh, _feature_angle)); }, py::arg("heh"), py::arg("feature_angle")=0.8) .def("calc_vertex_normal", [](Mesh& _self, OM::VertexHandle _vh) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } return vec2numpy(_self.calc_vertex_normal(_vh)); }) .def("calc_vertex_normal_fast", [](Mesh& _self, OM::VertexHandle _vh) { if (!_self.has_face_normals()) { _self.request_face_normals(); _self.update_face_normals(); } typename Mesh::Normal n; _self.calc_vertex_normal_fast(_vh, n); return vec2numpy(n); }) .def("calc_vertex_normal_correct", [](Mesh& _self, OM::VertexHandle _vh) { typename Mesh::Normal n; _self.calc_vertex_normal_correct(_vh, n); return vec2numpy(n); }) .def("calc_vertex_normal_loop", [](Mesh& _self, OM::VertexHandle _vh) { typename Mesh::Normal n; _self.calc_vertex_normal_loop(_vh, n); return vec2numpy(n); }) .def("calc_face_centroid", [](Mesh& _self, OM::FaceHandle _fh) { return vec2numpy(_self.calc_face_centroid(_fh)); }) .def("calc_edge_vector", [](Mesh& _self, OM::EdgeHandle _eh) { return vec2numpy(_self.calc_edge_vector(_eh)); }) .def("calc_edge_vector", [](Mesh& _self, OM::HalfedgeHandle _heh) { return vec2numpy(_self.calc_edge_vector(_heh)); }) .def("calc_sector_vectors", [](Mesh& _self, OM::HalfedgeHandle _heh) { typename Mesh::Normal vec0; typename Mesh::Normal vec1; _self.calc_sector_vectors(_heh, vec0, vec1); return std::make_tuple(vec2numpy(vec0), vec2numpy(vec1)); }) .def("calc_sector_normal", [](Mesh& _self, OM::HalfedgeHandle _heh) { typename Mesh::Normal n; _self.calc_sector_normal(_heh, n); return vec2numpy(n); }) //====================================================================== // numpy vector getter //====================================================================== .def("point", [](Mesh& _self, OM::VertexHandle _h) { return vec2numpy(_self, _self.point(_h)); }) .def("normal", [](Mesh& _self, OM::VertexHandle _h) { if (!_self.has_vertex_normals()) _self.request_vertex_normals(); return vec2numpy(_self, _self.normal(_h)); }) .def("normal", [](Mesh& _self, OM::HalfedgeHandle _h) { if (!_self.has_halfedge_normals()) _self.request_halfedge_normals(); return vec2numpy(_self, _self.normal(_h)); }) .def("normal", [](Mesh& _self, OM::FaceHandle _h) { if (!_self.has_face_normals()) _self.request_face_normals(); return vec2numpy(_self, _self.normal(_h)); }) .def("color", [](Mesh& _self, OM::VertexHandle _h) { if (!_self.has_vertex_colors()) _self.request_vertex_colors(); return vec2numpy(_self, _self.color(_h)); }) .def("color", [](Mesh& _self, OM::HalfedgeHandle _h) { if (!_self.has_halfedge_colors()) _self.request_halfedge_colors(); return vec2numpy(_self, _self.color(_h)); }) .def("color", [](Mesh& _self, OM::EdgeHandle _h) { if (!_self.has_edge_colors()) _self.request_edge_colors(); return vec2numpy(_self, _self.color(_h)); }) .def("color", [](Mesh& _self, OM::FaceHandle _h) { if (!_self.has_face_colors()) _self.request_face_colors(); return vec2numpy(_self, _self.color(_h)); }) .def("texcoord1D", [](Mesh& _self, OM::VertexHandle _h) { if (!_self.has_vertex_texcoords1D()) _self.request_vertex_texcoords1D(); return flt2numpy(_self, _self.texcoord1D(_h)); }) .def("texcoord1D", [](Mesh& _self, OM::HalfedgeHandle _h) { if (!_self.has_halfedge_texcoords1D()) _self.request_halfedge_texcoords1D(); return flt2numpy(_self, _self.texcoord1D(_h)); }) .def("texcoord2D", [](Mesh& _self, OM::VertexHandle _h) { if (!_self.has_vertex_texcoords2D()) _self.request_vertex_texcoords2D(); return vec2numpy(_self, _self.texcoord2D(_h)); }) .def("texcoord2D", [](Mesh& _self, OM::HalfedgeHandle _h) { if (!_self.has_halfedge_texcoords2D()) _self.request_halfedge_texcoords2D(); return vec2numpy(_self, _self.texcoord2D(_h)); }) .def("texcoord3D", [](Mesh& _self, OM::VertexHandle _h) { if (!_self.has_vertex_texcoords3D()) _self.request_vertex_texcoords3D(); return vec2numpy(_self, _self.texcoord3D(_h)); }) .def("texcoord3D", [](Mesh& _self, OM::HalfedgeHandle _h) { if (!_self.has_halfedge_texcoords3D()) _self.request_halfedge_texcoords3D(); return vec2numpy(_self, _self.texcoord3D(_h)); }) //====================================================================== // numpy vector setter //====================================================================== .def("set_point", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { _self.point(_h) = Point(_arr.at(0), _arr.at(1), _arr.at(2)); }) .def("set_normal", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { if (!_self.has_vertex_normals()) _self.request_vertex_normals(); _self.set_normal(_h, typename Mesh::Normal(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("set_normal", [](Mesh& _self, OM::HalfedgeHandle _h, py::array_t _arr) { if (!_self.has_halfedge_normals()) _self.request_halfedge_normals(); _self.set_normal(_h, typename Mesh::Normal(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("set_normal", [](Mesh& _self, OM::FaceHandle _h, py::array_t _arr) { if (!_self.has_face_normals()) _self.request_face_normals(); _self.set_normal(_h, typename Mesh::Normal(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("set_color", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { if(!_self.has_vertex_colors()) _self.request_vertex_colors(); _self.set_color(_h, typename Mesh::Color(_arr.at(0), _arr.at(1), _arr.at(2), _arr.at(3))); }) .def("set_color", [](Mesh& _self, OM::HalfedgeHandle _h, py::array_t _arr) { if(!_self.has_halfedge_colors()) _self.request_halfedge_colors(); _self.set_color(_h, typename Mesh::Color(_arr.at(0), _arr.at(1), _arr.at(2), _arr.at(3))); }) .def("set_color", [](Mesh& _self, OM::EdgeHandle _h, py::array_t _arr) { if(!_self.has_edge_colors()) _self.request_edge_colors(); _self.set_color(_h, typename Mesh::Color(_arr.at(0), _arr.at(1), _arr.at(2), _arr.at(3))); }) .def("set_color", [](Mesh& _self, OM::FaceHandle _h, py::array_t _arr) { if(!_self.has_face_colors()) _self.request_face_colors(); _self.set_color(_h, typename Mesh::Color(_arr.at(0), _arr.at(1), _arr.at(2), _arr.at(3))); }) .def("set_texcoord1D", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { if (!_self.has_vertex_texcoords1D()) _self.request_vertex_texcoords1D(); _self.set_texcoord1D(_h, _arr.at(0)); }) .def("set_texcoord1D", [](Mesh& _self, OM::HalfedgeHandle _h, py::array_t _arr) { if (!_self.has_halfedge_texcoords1D()) _self.request_halfedge_texcoords1D(); _self.set_texcoord1D(_h, _arr.at(0)); }) .def("set_texcoord2D", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { if (!_self.has_vertex_texcoords2D()) _self.request_vertex_texcoords2D(); _self.set_texcoord2D(_h, typename Mesh::TexCoord2D(_arr.at(0), _arr.at(1))); }) .def("set_texcoord2D", [](Mesh& _self, OM::HalfedgeHandle _h, py::array_t _arr) { if (!_self.has_halfedge_texcoords2D()) _self.request_halfedge_texcoords2D(); _self.set_texcoord2D(_h, typename Mesh::TexCoord2D(_arr.at(0), _arr.at(1))); }) .def("set_texcoord3D", [](Mesh& _self, OM::VertexHandle _h, py::array_t _arr) { if (!_self.has_vertex_texcoords3D()) _self.request_vertex_texcoords3D(); _self.set_texcoord3D(_h, typename Mesh::TexCoord3D(_arr.at(0), _arr.at(1), _arr.at(2))); }) .def("set_texcoord3D", [](Mesh& _self, OM::HalfedgeHandle _h, py::array_t _arr) { if (!_self.has_halfedge_texcoords3D()) _self.request_halfedge_texcoords3D(); _self.set_texcoord3D(_h, typename Mesh::TexCoord3D(_arr.at(0), _arr.at(1), _arr.at(2))); }) //====================================================================== // numpy matrix getter //====================================================================== .def("points", [](Mesh& _self) { return vec2numpy(_self, _self.point(OM::VertexHandle(0)), _self.n_vertices()); }) .def("vertex_normals", [](Mesh& _self) { if (!_self.has_vertex_normals()) _self.request_vertex_normals(); return vec2numpy(_self, _self.normal(OM::VertexHandle(0)), _self.n_vertices()); }) .def("vertex_colors", [](Mesh& _self) { if (!_self.has_vertex_colors()) _self.request_vertex_colors(); return vec2numpy(_self, _self.color(OM::VertexHandle(0)), _self.n_vertices()); }) .def("vertex_texcoords1D", [](Mesh& _self) { if (!_self.has_vertex_texcoords1D()) _self.request_vertex_texcoords1D(); return flt2numpy(_self, _self.texcoord1D(OM::VertexHandle(0)), _self.n_vertices()); }) .def("vertex_texcoords2D", [](Mesh& _self) { if (!_self.has_vertex_texcoords2D()) _self.request_vertex_texcoords2D(); return vec2numpy(_self, _self.texcoord2D(OM::VertexHandle(0)), _self.n_vertices()); }) .def("vertex_texcoords3D", [](Mesh& _self) { if (!_self.has_vertex_texcoords3D()) _self.request_vertex_texcoords3D(); return vec2numpy(_self, _self.texcoord3D(OM::VertexHandle(0)), _self.n_vertices()); }) .def("halfedge_normals", [](Mesh& _self) { if (!_self.has_halfedge_normals()) _self.request_halfedge_normals(); return vec2numpy(_self, _self.normal(OM::HalfedgeHandle(0)), _self.n_halfedges()); }) .def("halfedge_colors", [](Mesh& _self) { if (!_self.has_halfedge_colors()) _self.request_halfedge_colors(); return vec2numpy(_self, _self.color(OM::HalfedgeHandle(0)), _self.n_halfedges()); }) .def("halfedge_texcoords1D", [](Mesh& _self) { if (!_self.has_halfedge_texcoords1D()) _self.request_halfedge_texcoords1D(); return flt2numpy(_self, _self.texcoord1D(OM::HalfedgeHandle(0)), _self.n_halfedges()); }) .def("halfedge_texcoords2D", [](Mesh& _self) { if (!_self.has_halfedge_texcoords2D()) _self.request_halfedge_texcoords2D(); return vec2numpy(_self, _self.texcoord2D(OM::HalfedgeHandle(0)), _self.n_halfedges()); }) .def("halfedge_texcoords3D", [](Mesh& _self) { if (!_self.has_halfedge_texcoords3D()) _self.request_halfedge_texcoords3D(); return vec2numpy(_self, _self.texcoord3D(OM::HalfedgeHandle(0)), _self.n_halfedges()); }) .def("edge_colors", [](Mesh& _self) { if (!_self.has_edge_colors()) _self.request_edge_colors(); return vec2numpy(_self, _self.color(OM::EdgeHandle(0)), _self.n_edges()); }) .def("face_normals", [](Mesh& _self) { if (!_self.has_face_normals()) _self.request_face_normals(); return vec2numpy(_self, _self.normal(OM::FaceHandle(0)), _self.n_faces()); }) .def("face_colors", [](Mesh& _self) { if (!_self.has_face_colors()) _self.request_face_colors(); return vec2numpy(_self, _self.color (OM::FaceHandle(0)), _self.n_faces()); }) //====================================================================== // numpy indices //====================================================================== .def("edge_vertex_indices", &edge_vertex_indices) .def("ev_indices", &edge_vertex_indices) .def("halfedge_vertex_indices", &halfedge_vertex_indices) .def("hv_indices", &halfedge_vertex_indices) //====================================================================== // new property interface: single item //====================================================================== .def("vertex_property", &Mesh::template py_property) .def("halfedge_property", &Mesh::template py_property) .def("edge_property", &Mesh::template py_property) .def("face_property", &Mesh::template py_property) .def("set_vertex_property", &Mesh::template py_set_property) .def("set_halfedge_property", &Mesh::template py_set_property) .def("set_edge_property", &Mesh::template py_set_property) .def("set_face_property", &Mesh::template py_set_property) .def("has_vertex_property", &Mesh::template py_has_property) .def("has_halfedge_property", &Mesh::template py_has_property) .def("has_edge_property", &Mesh::template py_has_property) .def("has_face_property", &Mesh::template py_has_property) .def("remove_vertex_property", &Mesh::template py_remove_property) .def("remove_halfedge_property", &Mesh::template py_remove_property) .def("remove_edge_property", &Mesh::template py_remove_property) .def("remove_face_property", &Mesh::template py_remove_property) //====================================================================== // new property interface: generic //====================================================================== .def("vertex_property", &Mesh::template py_property_generic) .def("halfedge_property", &Mesh::template py_property_generic) .def("edge_property", &Mesh::template py_property_generic) .def("face_property", &Mesh::template py_property_generic) .def("set_vertex_property", &Mesh::template py_set_property_generic) .def("set_halfedge_property", &Mesh::template py_set_property_generic) .def("set_edge_property", &Mesh::template py_set_property_generic) .def("set_face_property", &Mesh::template py_set_property_generic) //====================================================================== // new property interface: array //====================================================================== .def("vertex_property_array", &Mesh::template py_property_array) .def("halfedge_property_array", &Mesh::template py_property_array) .def("edge_property_array", &Mesh::template py_property_array) .def("face_property_array", &Mesh::template py_property_array) .def("set_vertex_property_array", &Mesh::template py_set_property_array) .def("set_halfedge_property_array", &Mesh::template py_set_property_array) .def("set_edge_property_array", &Mesh::template py_set_property_array) .def("set_face_property_array", &Mesh::template py_set_property_array) //====================================================================== // new property interface: copy //====================================================================== .def("copy_property", &Mesh::template py_copy_property) .def("copy_property", &Mesh::template py_copy_property) .def("copy_property", &Mesh::template py_copy_property) .def("copy_property", &Mesh::template py_copy_property) ; expose_type_specific_functions(class_mesh); } #endif