latex-pstricks.cpp revision c751b68e56ace7b9d649ee20b7f25ed1f65922e3
/*
* LaTeX Printing
*
* Author:
* Michael Forbes <miforbes@mbhs.edu>
* Abhishek Sharma
*
* Copyright (C) 2004 Authors
*
* Released under GNU GPL, read the file 'COPYING' for more information
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <2geom/pathvector.h>
#include <2geom/sbasis-to-bezier.h>
#include <2geom/bezier-curve.h>
#include <2geom/hvlinesegment.h>
#include <errno.h>
#include <signal.h>
#include "util/units.h"
#include "helper/geom-curves.h"
#include "extension/print.h"
#include "extension/system.h"
#include "io/sys.h"
#include "latex-pstricks.h"
#include "sp-item.h"
#include "style.h"
#include "document.h"
#include <cstring>
namespace Inkscape {
namespace Extension {
namespace Internal {
PrintLatex::PrintLatex (void):
_width(0),
_height(0),
_stream(NULL)
{
}
PrintLatex::~PrintLatex (void)
{
if (_stream) fclose(_stream);
/* restore default signal handling for SIGPIPE */
#if !defined(_WIN32) && !defined(__WIN32__)
(void) signal(SIGPIPE, SIG_DFL);
#endif
return;
}
unsigned int PrintLatex::setup(Inkscape::Extension::Print * /*mod*/)
{
return TRUE;
}
unsigned int PrintLatex::begin (Inkscape::Extension::Print *mod, SPDocument *doc)
{
Inkscape::SVGOStringStream os;
int res;
FILE *osf = NULL;
const gchar * fn = NULL;
gsize bytesRead = 0;
gsize bytesWritten = 0;
GError* error = NULL;
os.setf(std::ios::fixed);
fn = mod->get_param_string("destination");
gchar* local_fn = g_filename_from_utf8( fn,
-1, &bytesRead, &bytesWritten, &error);
fn = local_fn;
/* TODO: Replace the below fprintf's with something that does the right thing whether in
* gui or batch mode (e.g. --print=blah). Consider throwing an exception: currently one of
* the callers (sp_print_document_to_file, "ret = mod->begin(doc)") wrongly ignores the
* return code.
*/
if (fn != NULL) {
while (isspace(*fn)) fn += 1;
Inkscape::IO::dump_fopen_call(fn, "K");
osf = Inkscape::IO::fopen_utf8name(fn, "w+");
if (!osf) {
fprintf(stderr, "inkscape: fopen(%s): %s\n", fn, strerror(errno));
g_free(local_fn);
return 0;
}
_stream = osf;
}
g_free(local_fn);
/* fixme: this is kinda icky */
#if !defined(_WIN32) && !defined(__WIN32__)
(void) signal(SIGPIPE, SIG_IGN);
#endif
res = fprintf(_stream, "%%LaTeX with PSTricks extensions\n");
/* flush this to test output stream as early as possible */
if (fflush(_stream)) {
/*g_print("caught error in sp_module_print_plain_begin\n");*/
if (ferror(_stream)) {
g_print("Error %d on output stream: %s\n", errno,
g_strerror(errno));
}
g_print("Printing failed\n");
/* fixme: should use pclose() for pipes */
fclose(_stream);
_stream = NULL;
fflush(stdout);
return 0;
}
// width and height in pt
_width = doc->getWidth() * Inkscape::Util::Quantity::convert(1, "px", "pt");
_height = doc->getHeight() * Inkscape::Util::Quantity::convert(1, "px", "pt");
if (res >= 0) {
os << "%%Creator: " << PACKAGE_STRING << "\n";
os << "%%Please note this file requires PSTricks extensions\n";
os << "\\psset{xunit=.5pt,yunit=.5pt,runit=.5pt}\n";
// from now on we can output px, but they will be treated as pt
os << "\\begin{pspicture}(" << doc->getWidth() << "," << doc->getHeight() << ")\n";
}
m_tr_stack.push( Geom::Scale(1, -1) * Geom::Translate(0, doc->getHeight())); /// @fixme hardcoded doc2dt transform
return fprintf(_stream, "%s", os.str().c_str());
}
unsigned int PrintLatex::finish(Inkscape::Extension::Print * /*mod*/)
{
if (_stream) {
fprintf(_stream, "\\end{pspicture}\n");
// Flush stream to be sure.
fflush(_stream);
fclose(_stream);
_stream = NULL;
}
return 0;
}
unsigned int PrintLatex::bind(Inkscape::Extension::Print * /*mod*/, Geom::Affine const &transform, float /*opacity*/)
{
if (!m_tr_stack.empty()) {
Geom::Affine tr_top = m_tr_stack.top();
m_tr_stack.push(transform * tr_top);
} else {
m_tr_stack.push(transform);
}
return 1;
}
unsigned int PrintLatex::release(Inkscape::Extension::Print * /*mod*/)
{
m_tr_stack.pop();
return 1;
}
unsigned int PrintLatex::comment(Inkscape::Extension::Print * /*mod*/,
const char * comment)
{
if (!_stream) {
return 0; // XXX: fixme, returning -1 as unsigned.
}
return fprintf(_stream, "%%! %s\n",comment);
}
unsigned int PrintLatex::fill(Inkscape::Extension::Print * /*mod*/,
Geom::PathVector const &pathv, Geom::Affine const &transform, SPStyle const *style,
Geom::OptRect const & /*pbox*/, Geom::OptRect const & /*dbox*/, Geom::OptRect const & /*bbox*/)
{
if (!_stream) {
return 0; // XXX: fixme, returning -1 as unsigned.
}
if (style->fill.isColor()) {
Inkscape::SVGOStringStream os;
float rgb[3];
float fill_opacity;
os.setf(std::ios::fixed);
fill_opacity=SP_SCALE24_TO_FLOAT(style->fill_opacity.value);
sp_color_get_rgb_floatv(&style->fill.value.color, rgb);
os << "{\n\\newrgbcolor{curcolor}{" << rgb[0] << " " << rgb[1] << " " << rgb[2] << "}\n";
os << "\\pscustom[linestyle=none,fillstyle=solid,fillcolor=curcolor";
if (fill_opacity!=1.0) {
os << ",opacity="<<fill_opacity;
}
os << "]\n{\n";
print_pathvector(os, pathv, transform);
os << "}\n}\n";
fprintf(_stream, "%s", os.str().c_str());
}
return 0;
}
unsigned int PrintLatex::stroke(Inkscape::Extension::Print * /*mod*/,
Geom::PathVector const &pathv, Geom::Affine const &transform, SPStyle const *style,
Geom::OptRect const & /*pbox*/, Geom::OptRect const & /*dbox*/, Geom::OptRect const & /*bbox*/)
{
if (!_stream) {
return 0; // XXX: fixme, returning -1 as unsigned.
}
if (style->stroke.isColor()) {
Inkscape::SVGOStringStream os;
float rgb[3];
float stroke_opacity;
Geom::Affine tr_stack = m_tr_stack.top();
double const scale = tr_stack.descrim();
os.setf(std::ios::fixed);
stroke_opacity=SP_SCALE24_TO_FLOAT(style->stroke_opacity.value);
sp_color_get_rgb_floatv(&style->stroke.value.color, rgb);
os << "{\n\\newrgbcolor{curcolor}{" << rgb[0] << " " << rgb[1] << " " << rgb[2] << "}\n";
os << "\\pscustom[linewidth=" << style->stroke_width.computed*scale<< ",linecolor=curcolor";
if (stroke_opacity!=1.0) {
os<<",strokeopacity="<<stroke_opacity;
}
if (style->stroke_dasharray_set &&
style->stroke_dash.n_dash &&
style->stroke_dash.dash) {
int i;
os << ",linestyle=dashed,dash=";
for (i = 0; i < style->stroke_dash.n_dash; i++) {
if ((i)) {
os << " ";
}
os << style->stroke_dash.dash[i];
}
}
os <<"]\n{\n";
print_pathvector(os, pathv, transform);
os << "}\n}\n";
fprintf(_stream, "%s", os.str().c_str());
}
return 0;
}
// FIXME: why is 'transform' argument not used?
void
PrintLatex::print_pathvector(SVGOStringStream &os, Geom::PathVector const &pathv_in, const Geom::Affine & /*transform*/)
{
if (pathv_in.empty())
return;
// Geom::Affine tf=transform; // why was this here?
Geom::Affine tf_stack=m_tr_stack.top(); // and why is transform argument not used?
Geom::PathVector pathv = pathv_in * tf_stack; // generates new path, which is a bit slow, but this doesn't have to be performance optimized
os << "\\newpath\n";
for(Geom::PathVector::const_iterator it = pathv.begin(); it != pathv.end(); ++it) {
os << "\\moveto(" << it->initialPoint()[Geom::X] << "," << it->initialPoint()[Geom::Y] << ")\n";
for(Geom::Path::const_iterator cit = it->begin(); cit != it->end_open(); ++cit) {
print_2geomcurve(os, *cit);
}
if (it->closed()) {
os << "\\closepath\n";
}
}
}
void
PrintLatex::print_2geomcurve(SVGOStringStream &os, Geom::Curve const &c)
{
using Geom::X;
using Geom::Y;
if( is_straight_curve(c) )
{
os << "\\lineto(" << c.finalPoint()[X] << "," << c.finalPoint()[Y] << ")\n";
}
else if(Geom::CubicBezier const *cubic_bezier = dynamic_cast<Geom::CubicBezier const*>(&c)) {
std::vector<Geom::Point> points = cubic_bezier->points();
os << "\\curveto(" << points[1][X] << "," << points[1][Y] << ")("
<< points[2][X] << "," << points[2][Y] << ")("
<< points[3][X] << "," << points[3][Y] << ")\n";
}
else {
//this case handles sbasis as well as all other curve types
Geom::Path sbasis_path = Geom::cubicbezierpath_from_sbasis(c.toSBasis(), 0.1);
for(Geom::Path::iterator iter = sbasis_path.begin(); iter != sbasis_path.end(); ++iter) {
print_2geomcurve(os, *iter);
}
}
}
bool
PrintLatex::textToPath(Inkscape::Extension::Print * ext)
{
return ext->get_param_bool("textToPath");
}
#include "clear-n_.h"
void PrintLatex::init(void)
{
/* SVG in */
Inkscape::Extension::build_from_mem(
"<inkscape-extension xmlns=\"" INKSCAPE_EXTENSION_URI "\">\n"
"<name>" N_("LaTeX Print") "</name>\n"
"<id>" SP_MODULE_KEY_PRINT_LATEX "</id>\n"
"<param name=\"destination\" type=\"string\"></param>\n"
"<param name=\"textToPath\" type=\"boolean\">true</param>\n"
"<print/>\n"
"</inkscape-extension>", new PrintLatex());
}
} /* namespace Internal */
} /* namespace Extension */
} /* namespace Inkscape */
/*
Local Variables:
mode:c++
c-file-style:"stroustrup"
c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
indent-tabs-mode:nil
fill-column:99
End:
*/
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 :