mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-30 20:21:12 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			182 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			182 lines
		
	
	
	
		
			6.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of libigl, a simple c++ geometry processing library.
 | |
| //
 | |
| // Copyright (C) 2014 Alec Jacobson <alecjacobson@gmail.com>
 | |
| //
 | |
| // This Source Code Form is subject to the terms of the Mozilla Public License
 | |
| // v. 2.0. If a copy of the MPL was not distributed with this file, You can
 | |
| // obtain one at http://mozilla.org/MPL/2.0/.
 | |
| #include "writePLY.h"
 | |
| #include <vector>
 | |
| 
 | |
| #include <igl/ply.h>
 | |
| #include <vector>
 | |
| 
 | |
| namespace
 | |
| {
 | |
|   template <typename Scalar> int ply_type();
 | |
|   template <> int ply_type<char>(){ return PLY_CHAR; }
 | |
|   template <> int ply_type<short>(){ return PLY_SHORT; }
 | |
|   template <> int ply_type<int>(){ return PLY_INT; }
 | |
|   template <> int ply_type<unsigned char>(){ return PLY_UCHAR; }
 | |
|   template <> int ply_type<unsigned short>(){ return PLY_SHORT; }
 | |
|   template <> int ply_type<unsigned int>(){ return PLY_UINT; }
 | |
|   template <> int ply_type<float>(){ return PLY_FLOAT; }
 | |
|   template <> int ply_type<double>(){ return PLY_DOUBLE; }
 | |
| }
 | |
| 
 | |
| template <
 | |
|   typename DerivedV,
 | |
|   typename DerivedF,
 | |
|   typename DerivedN,
 | |
|   typename DerivedUV>
 | |
| IGL_INLINE bool igl::writePLY(
 | |
|   const std::string & filename,
 | |
|   const Eigen::MatrixBase<DerivedV> & V,
 | |
|   const Eigen::MatrixBase<DerivedF> & F,
 | |
|   const Eigen::MatrixBase<DerivedN> & N,
 | |
|   const Eigen::MatrixBase<DerivedUV> & UV,
 | |
|   const bool ascii)
 | |
| {
 | |
|   // Largely based on obj2ply.c
 | |
|   typedef typename DerivedV::Scalar VScalar;
 | |
|   typedef typename DerivedN::Scalar NScalar;
 | |
|   typedef typename DerivedUV::Scalar UVScalar;
 | |
|   typedef typename DerivedF::Scalar FScalar;
 | |
| 
 | |
|   typedef struct Vertex
 | |
|   {
 | |
|     VScalar x,y,z,w;          /* position */
 | |
|     NScalar nx,ny,nz;         /* surface normal */
 | |
|     UVScalar s,t;              /* texture coordinates */
 | |
|   } Vertex;
 | |
| 
 | |
|   typedef struct Face
 | |
|   {
 | |
|     unsigned char nverts;    /* number of vertex indices in list */
 | |
|     FScalar *verts;              /* vertex index list */
 | |
|   } Face;
 | |
| 
 | |
|   igl::ply::PlyProperty vert_props[] =
 | |
|   { /* list of property information for a vertex */
 | |
|     {"x", ply_type<VScalar>(), ply_type<VScalar>(),offsetof(Vertex,x),0,0,0,0},
 | |
|     {"y", ply_type<VScalar>(), ply_type<VScalar>(),offsetof(Vertex,y),0,0,0,0},
 | |
|     {"z", ply_type<VScalar>(), ply_type<VScalar>(),offsetof(Vertex,z),0,0,0,0},
 | |
|     {"nx",ply_type<NScalar>(), ply_type<NScalar>(),offsetof(Vertex,nx),0,0,0,0},
 | |
|     {"ny",ply_type<NScalar>(), ply_type<NScalar>(),offsetof(Vertex,ny),0,0,0,0},
 | |
|     {"nz",ply_type<NScalar>(), ply_type<NScalar>(),offsetof(Vertex,nz),0,0,0,0},
 | |
|     {"s", ply_type<UVScalar>(),ply_type<UVScalar>(),offsetof(Vertex,s),0,0,0,0},
 | |
|     {"t", ply_type<UVScalar>(),ply_type<UVScalar>(),offsetof(Vertex,t),0,0,0,0},
 | |
|   };
 | |
| 
 | |
|   igl::ply::PlyProperty face_props[] =
 | |
|   { /* list of property information for a face */
 | |
|     {"vertex_indices", ply_type<FScalar>(), ply_type<FScalar>(), 
 | |
|       offsetof(Face,verts), 1, PLY_UCHAR, PLY_UCHAR, offsetof(Face,nverts)},
 | |
|   };
 | |
|   const bool has_normals = N.rows() > 0;
 | |
|   const bool has_texture_coords = UV.rows() > 0;
 | |
|   std::vector<Vertex> vlist(V.rows());
 | |
|   std::vector<Face> flist(F.rows());
 | |
|   for(size_t i = 0;i<(size_t)V.rows();i++)
 | |
|   {
 | |
|     vlist[i].x = V(i,0);
 | |
|     vlist[i].y = V(i,1);
 | |
|     vlist[i].z = V(i,2);
 | |
|     if(has_normals)
 | |
|     {
 | |
|       vlist[i].nx = N(i,0);
 | |
|       vlist[i].ny = N(i,1);
 | |
|       vlist[i].nz = N(i,2);
 | |
|     }
 | |
|     if(has_texture_coords)
 | |
|     {
 | |
|       vlist[i].s = UV(i,0);
 | |
|       vlist[i].t = UV(i,1);
 | |
|     }
 | |
|   }
 | |
|   for(size_t i = 0;i<(size_t)F.rows();i++)
 | |
|   {
 | |
|     flist[i].nverts = F.cols();
 | |
|     flist[i].verts = new FScalar[F.cols()];
 | |
|     for(size_t c = 0;c<(size_t)F.cols();c++)
 | |
|     {
 | |
|       flist[i].verts[c] = F(i,c);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const char * elem_names[] = {"vertex","face"};
 | |
|   FILE * fp = fopen(filename.c_str(),"w");
 | |
|   if(fp==NULL)
 | |
|   {
 | |
|     return false;
 | |
|   }
 | |
|   igl::ply::PlyFile * ply = igl::ply::ply_write(fp, 2,elem_names,
 | |
|       (ascii ? PLY_ASCII : PLY_BINARY_LE));
 | |
|   if(ply==NULL)
 | |
|   {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   std::vector<igl::ply::PlyProperty> plist;
 | |
|   plist.push_back(vert_props[0]);
 | |
|   plist.push_back(vert_props[1]);
 | |
|   plist.push_back(vert_props[2]);
 | |
|   if (has_normals)
 | |
|   {
 | |
|     plist.push_back(vert_props[3]);
 | |
|     plist.push_back(vert_props[4]);
 | |
|     plist.push_back(vert_props[5]);
 | |
|   }
 | |
|   if (has_texture_coords)
 | |
|   {
 | |
|     plist.push_back(vert_props[6]);
 | |
|     plist.push_back(vert_props[7]);
 | |
|   }
 | |
|   ply_describe_element(ply, "vertex", V.rows(),plist.size(),
 | |
|     &plist[0]);
 | |
| 
 | |
|   ply_describe_element(ply, "face", F.rows(),1,&face_props[0]);
 | |
|   ply_header_complete(ply);
 | |
|   int native_binary_type = igl::ply::get_native_binary_type2();
 | |
|   ply_put_element_setup(ply, "vertex");
 | |
|   for(const auto v : vlist)
 | |
|   {
 | |
|     ply_put_element(ply, (void *) &v, &native_binary_type);
 | |
|   }
 | |
|   ply_put_element_setup(ply, "face");
 | |
|   for(const auto f : flist)
 | |
|   {
 | |
|     ply_put_element(ply, (void *) &f, &native_binary_type);
 | |
|   }
 | |
| 
 | |
|   ply_close(ply);
 | |
|   for(size_t i = 0;i<(size_t)F.rows();i++)
 | |
|   {
 | |
|     delete[] flist[i].verts;
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| template <
 | |
|   typename DerivedV,
 | |
|   typename DerivedF>
 | |
| IGL_INLINE bool igl::writePLY(
 | |
|   const std::string & filename,
 | |
|   const Eigen::MatrixBase<DerivedV> & V,
 | |
|   const Eigen::MatrixBase<DerivedF> & F,
 | |
|   const bool ascii)
 | |
| {
 | |
|   Eigen::Matrix<typename DerivedV::Scalar,Eigen::Dynamic,Eigen::Dynamic> N,UV;
 | |
|   return writePLY(filename,V,F,N,UV,ascii);
 | |
| }
 | |
| 
 | |
| #ifdef IGL_STATIC_LIBRARY
 | |
| // Explicit template instantiation
 | |
| // generated by autoexplicit.sh
 | |
| template bool igl::writePLY<Eigen::Matrix<float, -1, 3, 1, -1, 3>, Eigen::Matrix<int, -1, 3, 1, -1, 3> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::MatrixBase<Eigen::Matrix<float, -1, 3, 1, -1, 3> > const&, Eigen::MatrixBase<Eigen::Matrix<int, -1, 3, 1, -1, 3> > const&, bool);
 | |
| // generated by autoexplicit.sh
 | |
| template bool igl::writePLY<Eigen::Matrix<double, 8, 3, 0, 8, 3>, Eigen::Matrix<int, 12, 3, 0, 12, 3> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::MatrixBase<Eigen::Matrix<double, 8, 3, 0, 8, 3> > const&, Eigen::MatrixBase<Eigen::Matrix<int, 12, 3, 0, 12, 3> > const&, bool);
 | |
| template bool igl::writePLY<Eigen::Matrix<double, -1, -1, 0, -1, -1>, Eigen::Matrix<int, -1, -1, 0, -1, -1> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::MatrixBase<Eigen::Matrix<double, -1, -1, 0, -1, -1> > const&, Eigen::MatrixBase<Eigen::Matrix<int, -1, -1, 0, -1, -1> > const&, bool);
 | |
| template bool igl::writePLY<Eigen::Matrix<double, -1, 3, 0, -1, 3>, Eigen::Matrix<int, -1, 3, 0, -1, 3> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::MatrixBase<Eigen::Matrix<double, -1, 3, 0, -1, 3> > const&, Eigen::MatrixBase<Eigen::Matrix<int, -1, 3, 0, -1, 3> > const&, bool);
 | |
| template bool igl::writePLY<Eigen::Matrix<double, -1, 3, 1, -1, 3>, Eigen::Matrix<int, -1, 3, 1, -1, 3> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::MatrixBase<Eigen::Matrix<double, -1, 3, 1, -1, 3> > const&, Eigen::MatrixBase<Eigen::Matrix<int, -1, 3, 1, -1, 3> > const&, bool);
 | |
| #endif
 | 
