#ifndef __VCGLIB_EXPORTERU3D #define __VCGLIB_EXPORTERU3D #include #include #include #include #include #include #include "export_idtf.h" #include namespace vcg { namespace tri { namespace io { namespace u3dparametersclasses { struct IDTFConverterParameters { const QString _converter_loc; const QString _input_file; const QString _output_file; IDTFConverterParameters(const QString& converter_loc,const QString& input_file,const QString& output_file) :_converter_loc(converter_loc),_input_file(input_file),_output_file(output_file) { } }; struct Movie15Parameters { Movie15Parameters() { _campar = NULL; } //WARNING: in movie15 y-axis and z-axis have been inverted!!! class CameraParameters { public: CameraParameters() :_cam_fov_angle(0.0f),_cam_roll_angle(0.0f),_obj_to_cam_dir(vcg::Point3f(0.0f,0.0f,0.0f)),_obj_to_cam_dist(0.0f),_obj_bbox_diag(0.0f),_obj_pos(vcg::Point3f(0.0f,0.0f,0.0f)) { } CameraParameters(const vcg::Point3f& mesh_center,const float mesh_bbox_diag) :_cam_fov_angle(0.0f),_cam_roll_angle(0.0f),_obj_to_cam_dir(vcg::Point3f(0.0f,0.0f,mesh_bbox_diag)),_obj_to_cam_dist(0.0),_obj_pos(mesh_center),_obj_bbox_diag(mesh_bbox_diag) { } CameraParameters(const float cam_fov_angle,const float cam_roll_angle, const vcg::Point3f& obj_to_cam_dir,const float obj_to_cam_dist, const float obj_bbox_diag, const vcg::Point3f& obj_pos = vcg::Point3f(0.0f,0.0f,0.0f)) :_cam_fov_angle(cam_fov_angle),_cam_roll_angle(cam_roll_angle),_obj_to_cam_dir(obj_to_cam_dir),_obj_to_cam_dist(obj_to_cam_dist),_obj_pos(obj_pos),_obj_bbox_diag(obj_bbox_diag) { } float _cam_fov_angle; float _cam_roll_angle; vcg::Point3f _obj_to_cam_dir; float _obj_to_cam_dist; vcg::Point3f _obj_pos; float _obj_bbox_diag; }; CameraParameters* _campar; }; } namespace QtUtilityFunctions { static void splitFilePath(const QString& filepath,QStringList& trim_path) { QString file_uniformed = filepath; file_uniformed.replace(QString("\\"),QString("/")); trim_path = file_uniformed.split("/"); } static QString fileNameFromTrimmedPath(const QStringList& file_path) { if (file_path.size() > 0) return file_path.at(file_path.size() - 1); else return QString(); } static QString fileNameFromPath(const QString& filepath) { QStringList list; splitFilePath(filepath,list); return fileNameFromTrimmedPath(list); } static QString pathWithoutFileName(const QString& filepath) { QString tmp(filepath); tmp.remove(fileNameFromPath(filepath)); return tmp; } static QString fileExtension(const QString& filepath) { QStringList trim_list; splitFilePath(filepath,trim_list); QString file = fileNameFromTrimmedPath(trim_list); trim_list = file.split("."); return trim_list.at(trim_list.size() - 1); } } class TGA_Exporter { public: struct TGAHeader { unsigned char identsize; unsigned char colourmaptype; unsigned char imagetype; unsigned char colormapspecs[5]; short xstart; short ystart; short width; short height; unsigned char bits; unsigned char descriptor; }; static void convert(const QString& outfile,const QImage& im) { TGAHeader tga; tga.identsize = 0; tga.colourmaptype = 0; tga.imagetype = 2; memset(tga.colormapspecs,0,5); tga.xstart = (short) im.offset().x(); tga.ystart = (short) im.offset().y(); tga.height = (short) im.height(); tga.width = (short) im.width(); std::ofstream file(qPrintable(outfile),std::ofstream::binary); unsigned char* tmpchan; int totbyte; if (im.hasAlphaChannel()) { //is a 8-digits binary number code // always 0 0 | mirroring | bits //(future uses)| image | for alpha-channel //-------------------------------------------- // 7 6 | 5 4 | 3 2 1 0 //-------------------------------------------- // 0 0 | 1 0 | 1 0 0 0 tga.descriptor = (char) 40; tga.bits = (char) 32; } else { //is a 8-digits binary number code // always 0 0 | mirroring | bits //(future uses)| image | for alpha-channel //-------------------------------------------- // 7 6 | 5 4 | 3 2 1 0 //-------------------------------------------- // 0 0 | 1 0 | 0 0 0 0 tga.descriptor = (char) 32; tga.bits = (char) 24; } totbyte = tga.height * tga.width * (tga.bits / 8); if (im.hasAlphaChannel()) tmpchan = const_cast(im.bits()); else { tmpchan = new unsigned char[totbyte]; unsigned int ii = 0; while(ii < totbyte) { tmpchan[ii] = const_cast(im.bits())[ii + (ii/3)]; ++ii; } } file.write((char *) &tga,sizeof(tga)); file.write(reinterpret_cast(tmpchan),totbyte); file.close(); } }; template class ExporterU3D { public: enum U3DError { E_NOERROR, // 0 E_ABORTED_CONVERSION //1 }; static const char *ErrorMsg(int error) { static const char * dae_error_msg[] = { "No errors", "Conversion Process From Idtf intermediate file to U3D format aborted" }; if(error>1 || error<0) return "Unknown error"; else return dae_error_msg[error]; }; private: static int InvokeConverter(const u3dparametersclasses::IDTFConverterParameters& par) { QProcess p; QString convstring = par._converter_loc; convstring = convstring + " -en1 -input " + par._input_file + " -output " + par._output_file; qDebug("Starting converter %s", qPrintable(convstring)); p.setProcessChannelMode(QProcess::MergedChannels); p.start(convstring); //wait until the task has been completed bool t = p.waitForFinished(-1); p.close(); return (int) t; } static void SaveLatex(SaveMeshType& /*m*/,const QString& file,const u3dparametersclasses::Movie15Parameters& mov_par) { Output_File latex(file.toStdString() + ".tex"); QString u3df = file + ".u3d"; QStringList file_trim; QtUtilityFunctions::splitFilePath(u3df,file_trim); std::string u3d_final = QtUtilityFunctions::fileNameFromTrimmedPath(file_trim).toStdString(); latex.write(0,"\\begin{document}"); latex.write(0,"\\includemovie["); latex.write(1,"poster,"); latex.write(1,"toolbar, %same as `controls\'"); latex.write(1,"label=" + u3d_final + ","); latex.write(1,"text=(" + u3d_final + "),"); std::string cam_string; u3dparametersclasses::Movie15Parameters::CameraParameters* cam = mov_par._campar; if (cam != NULL) { cam_string = cam_string + "3Daac=" + TextUtility::nmbToStr(cam->_cam_fov_angle) + ", 3Droll=" + TextUtility::nmbToStr(cam->_cam_roll_angle) + ", 3Dc2c=" + TextUtility::nmbToStr(cam->_obj_to_cam_dir.X()) + " " + TextUtility::nmbToStr(cam->_obj_to_cam_dir.Z()) + " " + TextUtility::nmbToStr(cam->_obj_to_cam_dir.Y()) + ", 3Droo=" + TextUtility::nmbToStr(cam->_obj_to_cam_dist) + ", 3Dcoo=" + TextUtility::nmbToStr(-cam->_obj_pos.X()) + " " + TextUtility::nmbToStr(cam->_obj_pos.Z()) + " " + TextUtility::nmbToStr(cam->_obj_pos.Y()) + ","; latex.write(1,cam_string); } latex.write(1,"3Dlights=File,"); latex.write(0,"]{\\linewidth}{\\linewidth}{" + u3d_final + "}"); latex.write(0,"\\end{document}"); } static void convertTexturesFiles(SaveMeshType& m,const QString& file_path,QStringList& conv_file) { for(unsigned int ii = 0; ii < m.textures.size(); ++ii) { QString qtmp(m.textures[ii].c_str()); QString ext = QtUtilityFunctions::fileExtension(qtmp); if (ext.toLower() != "tga") { QImage img(qtmp); QString stmp; if ((file_path.at(file_path.length() - 1) != '/') || (file_path.at(file_path.length() - 1) != '\\')) stmp = file_path + QString("/"); else stmp = file_path; qtmp = stmp + qtmp.remove(ext) + "tga"; m.textures[ii] = qtmp.toStdString(); TGA_Exporter::convert(qtmp,img); conv_file.push_back(qtmp); } } } static void removeConvertedTexturesFiles(QStringList& conv_file) { for(unsigned int ii = 0;ii < conv_file.size();++ii) { QDir dir(QtUtilityFunctions::pathWithoutFileName(conv_file[ii])); dir.remove(QtUtilityFunctions::fileNameFromPath(conv_file[ii])); } } public: static int Save(SaveMeshType& m,const char* output_file,const char* conv_loc,const u3dparametersclasses::Movie15Parameters& mov_par,const int mask) { QString curr = QDir::currentPath(); QString out(output_file); QStringList out_trim; QtUtilityFunctions::splitFilePath(out,out_trim); QString tmp(QDir::tempPath()); tmp = tmp + "/" + QtUtilityFunctions::fileNameFromTrimmedPath(out_trim) + ".idtf"; QString conv_loc_st(conv_loc); QString output_file_st(output_file); //if there are textures file that aren't in tga format I have to convert them //I maintain the converted file name (i.e. file_path + originalname without extension + tga) in mesh.textures but I have to revert to the original ones //before the function return. QStringList oldtextname; for(unsigned int ii = 0; ii < m.textures.size();++ii) oldtextname.push_back(m.textures[ii].c_str()); //tmp vector to save the tga created files that should be deleted. QStringList convfile; convertTexturesFiles(m,curr,convfile); vcg::tri::io::ExporterIDTF::Save(m,qPrintable(tmp),mask); u3dparametersclasses::IDTFConverterParameters idtfpar(conv_loc_st,tmp,output_file_st); qDebug("conv_loc_st '%s'", qPrintable(conv_loc_st)); qDebug("conv_loc '%s'", conv_loc); qDebug("idtfpar._converter_loc '%s'", qPrintable(idtfpar._converter_loc)); int res = InvokeConverter(idtfpar); m.textures.clear(); for(QStringList::iterator it = oldtextname.begin(); it != oldtextname.end();++it) m.textures.push_back(it->toStdString()); //if some tga files have been created I have to delete them removeConvertedTexturesFiles(convfile); QDir::setCurrent(curr); QString lat (output_file); QStringList l = lat.split("."); SaveLatex(m,l[0],mov_par); QDir dir(QDir::tempPath()); dir.remove(tmp); if (res) return 0; else return 1; } static int GetExportMaskCapability() { int capability = 0; //vert capability |= Mask::IOM_VERTNORMAL; ////wedg capability |= Mask::IOM_WEDGTEXCOORD; capability |= Mask::IOM_WEDGNORMAL; return capability; } }; } } } #endif