/**************************************************************************** * VCGLib o o * * Visual and Computer Graphics Library o o * * _ O _ * * Copyright(C) 2006 \/)\/ * * Visual Computing Lab /\/| * * ISTI - Italian National Research Council | * * \ * * All rights reserved. * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This program 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 General Public License (http://www.gnu.org/licenses/gpl.txt) * * for more details. * * * ****************************************************************************/ #ifndef __VCG_LIB_EXPORTER_SVG #define __VCG_LIB_EXPORTER_SVG namespace vcg { namespace tri { namespace io { /** * SVG Properties. * * Support class to set the properties of the SVG exporter. */ class SVGProperties { // definitions public: // When multiple meshes are passed, they are arranged in a grid according these two values. // the default is two column and enough row. If numRow is not sufficient it is automatically enlarged. int numCol; int numRow; Point2f sizeCm; // The size, in the drawing, of each ViewBox (in cm) Point2f marginCm; // how much space between each slice box (in cm) Point2f pageSizeCm() // This is automatically computed from the above values { float xSize = numCol*sizeCm[0] + numCol*marginCm[0] + marginCm[0]; float ySize = numRow*sizeCm[1] + numRow*marginCm[1] + marginCm[1]; return Point2f(xSize,ySize); } Point3f projDir; // Direction of the Projection Point3f projUp; Point3f projCenter; // the 3d point that after projection will fall exactly in the center of the ViewBox. // How the mesh will be scaled. // if this value is 0 the bounding box of all the passed meshes will be used to compute the scale and center // otherwise it is a scaling factor that is used to place the mesh in a unit cube (-1..1) // usually it is 2/bbox.Diag float scale; // SVG Style Parameters int lineWidthPt; // Line width. std::string strokeColor; // Stroke color (see StrokeColor). std::string strokeLineCap;// Stroke linecap (see StrokeLineCap). //Text details bool showTextDetails; public: SVGProperties() { lineWidthPt = 1; strokeColor = "black"; strokeLineCap = "round"; // default projection (XZ plane with the z up) projDir= Point3f(0.0, 1.0, 0.0); projUp = Point3f(0.0, 0.0, 1.0); scale=0; //viewBox=Point2f(10, 10); projCenter=Point3f(0, 0, 0); sizeCm=Point2f(10,10); marginCm=Point2f(1,1); showTextDetails=true; numCol=2; numRow=10; } }; /** * SVG exporter. * * This exporter save a mesh of EdgeMesh type in the SVG format. * Most of the features of the SVG format are not supported. * The given EdgeMesh is saved as a set lines. The properties * of the SVG export can be set through the SVGProp class. */ template <class EdgeMeshType> class ExporterSVG { public: // Standard saving Function // just a wrapper to the below static bool Save(EdgeMeshType &m, const char *filename, SVGProperties & pro) { std::vector<EdgeMeshType*> MeshVec; MeshVec.push_back(&m); return Save(MeshVec,filename,pro); } // Main saving function // save a Multiple Set of Edge Meshes on a single SVG files static bool Save(std::vector<EdgeMeshType*> &meshVec, const char *filename, SVGProperties & pro) { FILE * fpo = fopen(filename,"w"); if (fpo==NULL) return false; WriteXmlHead(fpo, pro); for(size_t i=0;i<meshVec.size();++i) { WriteXmlBody(fpo, *meshVec[i], pro, i ); } fprintf(fpo, "</svg>"); fclose(fpo); return true; } static void WriteXmlHead(FILE *o, SVGProperties & pro) { fprintf(o, "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>\n"); fprintf(o, "<!-- Created with vcg library -->\n"); fprintf(o, "<svg width=\"%fcm\" height=\"%fcm\" \n",pro.pageSizeCm()[0], pro.pageSizeCm()[1]); fprintf(o, " xmlns=\"http://www.w3.org/2000/svg\" \n"); fprintf(o, " xmlns:xlink=\"http://www.w3.org/1999/xlink\" \n"); fprintf(o, " xmlns:dc=\"http://purl.org/dc/elements/1.1/\" \n"); fprintf(o, " xmlns:cc=\"http://web.resource.org/cc/\" \n"); fprintf(o, " xmlns:rdf=\"http://www.w3.org/1999/02/22-rdf-syntax-ns#\" \n"); fprintf(o, " xmlns:svg=\"http://www.w3.org/2000/svg\" \n \n"); fprintf(o, "id=\"svg2\"> \n"); fprintf(o, " <defs id=\"defs4\"/> \n"); fprintf(o, " <metadata id=\"metadata7\"> \n"); fprintf(o, " <rdf:RDF> \n"); fprintf(o, " <cc:Work rdf:about=\"\"> \n"); fprintf(o, " <dc:format>image/svg+xml</dc:format> \n"); fprintf(o, " <dc:type rdf:resource=\"http://purl.org/dc/dcmitype/StillImage\" /> \n"); fprintf(o, " </cc:Work> \n"); fprintf(o, " </rdf:RDF> \n"); fprintf(o, "</metadata> \n \n"); } static void WriteXmlBody(FILE* fpo, EdgeMeshType &mp, SVGProperties &pro, int meshIndex) { int rowInd = meshIndex / pro.numCol; int colInd = meshIndex % pro.numCol; fprintf(fpo, " <rect width= \" %fcm \" height= \" %fcm \" x=\"%fcm \" y=\"%fcm \" " " style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" /> \n", pro.sizeCm[0], pro.sizeCm[1], pro.marginCm[0]+colInd*(pro.sizeCm[0]+pro.marginCm[0]), pro.marginCm[1]+rowInd*(pro.sizeCm[1]+pro.marginCm[1])); fprintf(fpo, "<g stroke=\"%s\" stroke-linecap=\"%s\" stroke-width = \"%fpt\" > \n", pro.strokeColor.c_str(), pro.strokeLineCap.c_str(),pro.lineWidthPt/100.0f); fprintf(fpo, " <svg id = \"SliceNum%d\" viewBox=\"-1000 -1000 2000 2000\" width=\"%fcm\" height=\"%fcm\" x=\"%fcm\" y=\"%fcm\" >\n", meshIndex,pro.sizeCm[0],pro.sizeCm[1], pro.marginCm[0]+colInd*(pro.sizeCm[0]+pro.marginCm[0]), pro.marginCm[1]+rowInd*(pro.sizeCm[1]+pro.marginCm[1]) ); // Main loop of edge printing typename EdgeMeshType::EdgeIterator i; // XY projection. // It is a classcial ortho projection // eg it resolves to a rotation Matrix such that // - the passed projDir become the z axis // - the passed projUp lie on the upper YZ plane. // First Step align projDir to Z Matrix33f rotM = RotationMatrix(pro.projDir,Point3f(0,0,1),false); Point3f rotatedUp = rotM * pro.projUp; Point3f rotCenter = rotM * pro.projCenter; float scale = pro.scale; if(scale==0) scale = 2.0/mp.bbox.Diag(); for (i = mp.edge.begin(); i != mp.edge.end(); ++i) if(!(*i).IsD()) { Point3f p0 = (-rotCenter + rotM * ((*i).V(0)->P()))*scale*1000; Point3f p1 = (-rotCenter + rotM * ((*i).V(1)->P()))*scale*1000; fprintf(fpo, " <line x1=\"%f\" y1=\"%f\" x2=\"%f\" y2=\"%f\" />\n", p0[0],p0[1],p1[0],p1[1]); } fprintf(fpo, " </svg>\n"); fprintf(fpo, "</g>\n"); } }; }; // namespace io }; // namespace edge }; // namespace vcg #endif // __VCG_LIB_EXPORTER_SVG