/*************************************************************************** * Copyright (c) 2017 Lorenz Lechner * * * * This file is part of the FreeCAD CAx development system. * * * * This library is free software; you can redistribute it and/or * * modify it under the terms of the GNU Library General Public * * License as published by the Free Software Foundation; either * * version 2 of the License, or (at your option) any later version. * * * * This library is distributed in the hope that it will be useful, * * but WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU Library General Public License for more details. * * * * You should have received a copy of the GNU Library General Public * * License along with this library; see the file COPYING.LIB. If not, * * write to the Free Software Foundation, Inc., 59 Temple Place, * * Suite 330, Boston, MA 02111-1307, USA * * * ***************************************************************************/ #include "PreCompiled.h" #ifndef _PreComp_ # include # include # include # include # include # include # include #endif // boost is purposely not in the precompiled headers, see // https://github.com/FreeCAD/FreeCAD/pull/7979#issuecomment-1358123252 #include #include #include #include #include #include #include #include #include #include #include #include #include #include "MeshFlattening.h" #include "MeshFlatteningLscmRelax.h" #include "MeshFlatteningNurbs.h" namespace py = boost::python; const TopoDS_Face& getTopoDSFace(const py::object& face) { if (PyObject_TypeCheck(face.ptr(), &(Part::TopoShapeFacePy::Type))) { const Part::TopoShapeFacePy* fpy = static_cast(face.ptr()); const TopoDS_Face& myFace = TopoDS::Face(fpy->getTopoShapePtr()->getShape()); return myFace; } throw std::invalid_argument("must be a face"); } const TopoDS_Edge& getTopoDSEdge(py::object* edge) { if (PyObject_TypeCheck(edge->ptr(), &(Part::TopoShapeEdgePy::Type))) { const Part::TopoShapeEdgePy* epy = static_cast(edge->ptr()); const TopoDS_Edge& myEdge = TopoDS::Edge(epy->getTopoShapePtr()->getShape()); return myEdge; } throw std::invalid_argument("must be an edge"); } Py::Object makeEdge(const TopoDS_Edge& edge) { return Py::asObject(new Part::TopoShapeEdgePy(new Part::TopoShape(edge))); } std::shared_ptr FaceUnwrapper_face(const py::object& face) { const TopoDS_Face& myFace = getTopoDSFace(face); return std::make_shared(myFace); } std::shared_ptr FaceUnwrapper_mesh(const py::object& points, const py::object& facets) { try { Py::Sequence l1(points.ptr()); ColMat coords; coords.resize(l1.size(), 3); int row = 0; for (Py::Sequence::iterator it = l1.begin(); it != l1.end(); ++it, ++row) { Py::Sequence c(*it); int col = 0; for (Py::Sequence::iterator jt = c.begin(); jt != c.end(); ++jt, ++col) { double v = static_cast(Py::Float(*jt)); coords(row, col) = v; } } Py::Sequence l2(facets.ptr()); ColMat triangles; triangles.resize(l2.size(), 3); row = 0; for (Py::Sequence::iterator it = l2.begin(); it != l2.end(); ++it, ++row) { Py::Sequence c(*it); int col = 0; for (Py::Sequence::iterator jt = c.begin(); jt != c.end(); ++jt, ++col) { long v = static_cast(Py::Long(*jt)); triangles(row, col) = v; } } return std::shared_ptr(new FaceUnwrapper(coords, triangles)); } catch (const Py::Exception&) { Base::PyException e; throw std::invalid_argument(e.what()); } } boost::python::list interpolateFlatFacePy(FaceUnwrapper& instance, const py::object& face) { const TopoDS_Face& myFace = getTopoDSFace(face); ColMat mat = instance.interpolateFlatFace(myFace); boost::python::list plist; auto cols = mat.cols(); auto rows = mat.rows(); for (int i=0; i> mat_array = instance.getFlatBoundaryNodes(); boost::python::list ary; for (auto& mat : mat_array) { boost::python::list plist; auto cols = mat.cols(); auto rows = mat.rows(); for (int i=0; i struct eigen_matrix { static PyObject* convert(const eigen_type& mat) { // py::list ary; for (int i = 0; i < mat.rows(); i++) { py::list row; for (int j = 0; j < mat.cols(); j++) { row.append(mat.coeff(i ,j)); } ary.append(row); } return boost::python::incref(ary.ptr()); } static void to_python_converter() { py::to_python_converter>(); } }; } // namespace fm BOOST_PYTHON_MODULE(flatmesh) { //m.doc() = "functions to unwrapp faces/ meshes"; py::class_("LscmRelax") .def(py::init, ColMat, std::vector>()) .def("lscm", &lscmrelax::LscmRelax::lscm) .def("relax", &lscmrelax::LscmRelax::relax) .def("rotate_by_min_bound_area", &lscmrelax::LscmRelax::rotate_by_min_bound_area) .def("transform", &lscmrelax::LscmRelax::transform) .def_readonly("rhs", &lscmrelax::LscmRelax::rhs) .def_readonly("MATRIX", &lscmrelax::LscmRelax::MATRIX) .def_readonly("area", &lscmrelax::LscmRelax::get_area) .def_readonly("flat_area", &lscmrelax::LscmRelax::get_flat_area) // .def_readonly("flat_vertices", [](lscmrelax::LscmRelax& L){return L.flat_vertices.transpose();}, py::return_value_policy()) .def_readonly("flat_vertices_3D", &lscmrelax::LscmRelax::get_flat_vertices_3D); py::class_("NurbsBase2D") .def(py::init()) .def_readonly("u_knots", &nurbs::NurbsBase2D::u_knots) .def_readonly("weights", &nurbs::NurbsBase2D::weights) .def_readonly("degree_u", &nurbs::NurbsBase2D::degree_u) // .def_readonly("v_knots", &nurbs::NurbsBase2D::u_knots) .def_readonly("degree_v", &nurbs::NurbsBase2D::degree_u) .def("getUVMesh", &nurbs::NurbsBase2D::getUVMesh) .def("computeFirstDerivatives", &nurbs::NurbsBase2D::computeFirstDerivatives) .def("getInfluenceVector", &nurbs::NurbsBase2D::getInfluenceVector) .def("getInfluenceMatrix", &nurbs::NurbsBase2D::getInfluenceMatrix) .def("getDuVector", &nurbs::NurbsBase2D::getDuVector) .def("getDuMatrix", &nurbs::NurbsBase2D::getDuMatrix) .def("getDvVector", &nurbs::NurbsBase2D::getDvVector) .def("getDvMatrix", &nurbs::NurbsBase2D::getDvMatrix) .def("interpolateUBS", &nurbs::NurbsBase2D::interpolateUBS); py::class_("NurbsBase1D") .def(py::init()) .def_readonly("u_knots", &nurbs::NurbsBase1D::u_knots) .def_readonly("weights", &nurbs::NurbsBase1D::weights) .def_readonly("degree_u", &nurbs::NurbsBase1D::degree_u) .def("getUMesh", &nurbs::NurbsBase1D::getUMesh) .def("computeFirstDerivatives", &nurbs::NurbsBase1D::computeFirstDerivatives) .def("getInfluenceVector", &nurbs::NurbsBase1D::getInfluenceVector) .def("getInfluenceMatrix", &nurbs::NurbsBase1D::getInfluenceMatrix) .def("getDuVector", &nurbs::NurbsBase1D::getDuVector) .def("getDuMatrix", &nurbs::NurbsBase1D::getDuMatrix) .add_static_property("getKnotSequence", &nurbs::NurbsBase1D::getKnotSequence) .add_static_property("getWeightList", &nurbs::NurbsBase1D::getWeightList); py::class_("FaceUnwrapper") .def("__init__", py::make_constructor(&FaceUnwrapper_face)) .def("__init__", py::make_constructor(&FaceUnwrapper_mesh)) .def(py::init, ColMat>()) .def("findFlatNodes", &FaceUnwrapper::findFlatNodes) .def("interpolateFlatFace", &interpolateFlatFacePy) .def("getFlatBoundaryNodes", &getFlatBoundaryNodesPy) .add_property("tris", py::make_getter(&FaceUnwrapper::tris, py::return_value_policy())) .add_property("nodes", py::make_getter(&FaceUnwrapper::xyz_nodes, py::return_value_policy())) .add_property("uv_nodes", py::make_getter(&FaceUnwrapper::uv_nodes, py::return_value_policy())) .add_property("ze_nodes", py::make_getter(&FaceUnwrapper::ze_nodes, py::return_value_policy())) .add_property("ze_poles", py::make_getter(&FaceUnwrapper::ze_poles, py::return_value_policy())) .add_property("A", py::make_getter(&FaceUnwrapper::A, py::return_value_policy())); fm::eigen_matrix::to_python_converter(); fm::eigen_matrix>::to_python_converter(); fm::eigen_matrix>::to_python_converter(); fm::eigen_matrix>::to_python_converter(); fm::eigen_matrix>::to_python_converter(); }