vcglib/wrap/io_edgemesh/export_svg.h

280 lines
11 KiB
C++

/****************************************************************************
* 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 edg
{
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;
bool crossHairs; //toggle crossHairs printing
float crossHairRad; //crossHair radius in cm
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;
crossHairs=true;
crossHairRad=.25f;
}
};
/**
* 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(int 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]));
// write crosshairs
if (pro.crossHairs){
// WriteCrossHairs(fpo);
float r=pro.crossHairRad;
//ch1
fprintf(fpo, " <svg id = \"ch1%d\" viewBox=\"0,0,100,100\" x=\"%fcm\" y=\"%fcm\" width=\"%fcm\" height=\"%fcm\" >\n",meshIndex,
pro.marginCm[0]+colInd*(pro.sizeCm[0]+pro.marginCm[0]),
pro.marginCm[1]+rowInd*(pro.sizeCm[1]+pro.marginCm[1]),
r*2,r*2
);
fprintf(fpo," <ellipse cx=\"50\" cy=\"50\" rx=\"50\" ry=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"50\" y1=\"0\" x2=\"50\" y2=\"100\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"0\" y1=\"50\" x2=\"100\" y2=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," </svg>");
//ch2
fprintf(fpo, " <svg id = \"ch1%d\" viewBox=\"0,0,100,100\" x=\"%fcm\" y=\"%fcm\" width=\"%fcm\" height=\"%fcm\" >\n",meshIndex,
pro.marginCm[0]+(colInd+1)*(pro.sizeCm[0]+pro.marginCm[0])-r*2-pro.marginCm[0],
pro.marginCm[1]+(rowInd+1)*(pro.sizeCm[1]+pro.marginCm[1])-r*2-pro.marginCm[1],
r*2,r*2
);
fprintf(fpo," <ellipse cx=\"50\" cy=\"50\" rx=\"50\" ry=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"50\" y1=\"0\" x2=\"50\" y2=\"100\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"0\" y1=\"50\" x2=\"100\" y2=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," </svg>");
//ch3
fprintf(fpo, " <svg id = \"ch1%d\" viewBox=\"0,0,100,100\" x=\"%fcm\" y=\"%fcm\" width=\"%fcm\" height=\"%fcm\" >\n",meshIndex,
pro.marginCm[0]+colInd*(pro.sizeCm[0]+pro.marginCm[0]),
pro.marginCm[1]+(rowInd+1)*(pro.sizeCm[1]+pro.marginCm[1])-pro.marginCm[1]-2*r,
r*2,r*2
);
fprintf(fpo," <ellipse cx=\"50\" cy=\"50\" rx=\"50\" ry=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"50\" y1=\"0\" x2=\"50\" y2=\"100\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"0\" y1=\"50\" x2=\"100\" y2=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," </svg>");
//ch3
fprintf(fpo, " <svg id = \"ch1%d\" viewBox=\"0,0,100,100\" x=\"%fcm\" y=\"%fcm\" width=\"%fcm\" height=\"%fcm\" >\n",meshIndex,
pro.marginCm[0]+(colInd+1)*(pro.sizeCm[0]+pro.marginCm[0])-pro.marginCm[0]-2*r,
pro.marginCm[1]+rowInd*(pro.sizeCm[1]+pro.marginCm[1]),
r*2,r*2
);
fprintf(fpo," <ellipse cx=\"50\" cy=\"50\" rx=\"50\" ry=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"50\" y1=\"0\" x2=\"50\" y2=\"100\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," <line x1=\"0\" y1=\"50\" x2=\"100\" y2=\"50\" style= \" stroke-width:1pt; fill-opacity:0.0; stroke:rgb(0,0,0)\" />");
fprintf(fpo," </svg>");
}
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.edges.begin(); i != mp.edges.end(); ++i)
{
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