javafx-out.cpp revision 41c1aa8f1639eadc4b1599d72380c061f7f82987
/*
* A simple utility for exporting Inkscape svg Shapes as JavaFX paths.
*
* For information on the JavaFX file format, see:
* https://openjfx.dev.java.net/
*
* Authors:
* Bob Jamison <ishmal@inkscape.org>
*
* Copyright (C) 2008 Authors
*
* Released under GNU GPL, read the file 'COPYING' for more information
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "javafx-out.h"
#include <inkscape.h>
#include <inkscape_version.h>
#include <sp-path.h>
#include <sp-linear-gradient.h>
#include <sp-radial-gradient.h>
#include <style.h>
#include <display/curve.h>
#include <svg/svg.h>
#include <extension/system.h>
#include <2geom/pathvector.h>
#include <2geom/rect.h>
#include <2geom/bezier-curve.h>
#include <2geom/hvlinesegment.h>
#include "helper/geom.h"
#include "helper/geom-curves.h"
#include <io/sys.h>
#include <string>
#include <stdio.h>
#include <stdarg.h>
namespace Inkscape
{
namespace Extension
{
namespace Internal
{
//########################################################################
//# M E S S A G E S
//########################################################################
static void err(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
g_logv(NULL, G_LOG_LEVEL_WARNING, fmt, args);
va_end(args);
}
//########################################################################
//# U T I L I T Y
//########################################################################
/**
* Got this method from Bulia, and modified it a bit. It basically
* starts with this style, gets its SPObject parent, walks up the object
* tree and finds all of the opacities and multiplies them.
*
* We use this for our "flat" object output. If the code is modified
* to reflect a tree of <groups>, then this will be unneccessary.
*/
static double effective_opacity(const SPStyle *style)
{
double val = 1.0;
for (SPObject const *obj = style->object; obj ; obj = obj->parent)
{
style = SP_OBJECT_STYLE(obj);
if (style)
val *= SP_SCALE24_TO_FLOAT(style->opacity.value);
}
return val;
}
//########################################################################
//# OUTPUT FORMATTING
//########################################################################
/**
* We want to control floating output format
*/
static JavaFXOutput::String dstr(double d)
{
char dbuf[G_ASCII_DTOSTR_BUF_SIZE+1];
g_ascii_formatd(dbuf, G_ASCII_DTOSTR_BUF_SIZE,
"%.8f", (gdouble)d);
JavaFXOutput::String s = dbuf;
return s;
}
/**
* Format an rgba() string
*/
static JavaFXOutput::String rgba(guint32 rgba)
{
double r = SP_RGBA32_R_F(rgba);
double g = SP_RGBA32_G_F(rgba);
double b = SP_RGBA32_B_F(rgba);
double a = SP_RGBA32_A_F(rgba);
char buf[128];
snprintf(buf, 79, "Color.color(%s, %s, %s, %s)",
dstr(r).c_str(), dstr(g).c_str(), dstr(b).c_str(), dstr(a).c_str());
JavaFXOutput::String s = buf;
return s;
}
/**
* Format an rgba() string for a color and a 0.0-1.0 alpha
*/
static JavaFXOutput::String rgba(SPColor color, gdouble alpha)
{
return rgba(color.toRGBA32(alpha));
}
/**
* Output data to the buffer, printf()-style
*/
void JavaFXOutput::out(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
gchar *output = g_strdup_vprintf(fmt, args);
va_end(args);
outbuf.append(output);
g_free(output);
}
/**
* Output header data to the buffer, printf()-style
*/
void JavaFXOutput::fout(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
gchar *output = g_strdup_vprintf(fmt, args);
va_end(args);
foutbuf.append(output);
g_free(output);
}
/**
* Output the file header
*/
bool JavaFXOutput::doHeader()
{
time_t tim = time(NULL);
out("/*###################################################################\n");
out("### This JavaFX document was generated by Inkscape\n");
out("### http://www.inkscape.org\n");
out("### Created: %s", ctime(&tim));
out("### Version: %s\n", INKSCAPE_VERSION);
out("#####################################################################\n");
out("### NOTES:\n");
out("### ============\n");
out("### JavaFX information can be found at\n");
out("### hhttps://openjfx.dev.java.net\n");
out("###\n");
out("### If you have any problems with this output, please see the\n");
out("### Inkscape project at http://www.inkscape.org, or visit\n");
out("### the #inkscape channel on irc.freenode.net . \n");
out("###\n");
out("###################################################################*/\n");
out("\n\n");
out("/*###################################################################\n");
out("## Exports in this file\n");
out("##==========================\n");
out("## Shapes : %d\n", nrShapes);
out("## Nodes : %d\n", nrNodes);
out("###################################################################*/\n");
out("\n\n");
out("import javafx.application.*;\n");
out("import javafx.scene.*;\n");
out("import javafx.scene.geometry.*;\n");
out("import javafx.scene.paint.*;\n");
out("import javafx.scene.transform.*;\n");
out("\n");
out("import java.lang.System;\n");
out("\n\n");
out("public class %s extends CustomNode {\n", name.c_str());
out("\n\n");
outbuf.append(foutbuf);
out("\n\n");
out("public function create() : Node\n");
out("{\n");
out("return Group\n");
out(" {\n");
out(" content:\n");
out(" [\n");
return true;
}
/**
* Output the file footer
*/
bool JavaFXOutput::doTail()
{
int border = 25.0;
out(" ] // content\n");
out(" transform: [ Translate.translate(%s, %s), ]\n",
dstr((-minx) + border).c_str(), dstr((-miny) + border).c_str());
out(" } // Group\n");
out("} // end function create()\n");
out("\n\n");
out("}\n");
out("/*###################################################################\n");
out("### E N D C L A S S %s\n", name.c_str());
out("###################################################################*/\n");
out("\n\n\n\n");
out("\n");
out("Frame {\n");
out(" title: \"TestFrame\"\n");
out(" width: (%s).intValue()\n", dstr(maxx-minx + border * 2.0).c_str());
out(" height: (%s).intValue()\n", dstr(maxy-miny + border * 2.0).c_str());
out(" closeAction: function()\n");
out(" {\n");
out(" System.exit( 0 );\n");
out(" }\n");
out(" visible: true\n");
out(" stage: Stage {\n");
out(" content: %s{}\n", name.c_str());
out(" }\n");
out("}\n");
out("\n\n");
out("/*###################################################################\n");
out("### E N D C L A S S %s\n", name.c_str());
out("###################################################################*/\n");
out("\n\n");
return true;
}
/**
* Output gradient information to the buffer
*/
bool JavaFXOutput::doGradient(SPGradient *grad, const String &id)
{
if (SP_IS_LINEARGRADIENT(grad))
{
SPLinearGradient *g = SP_LINEARGRADIENT(grad);
fout("function %s() : LinearGradient\n", id.c_str());
fout("{\n");
fout(" return LinearGradient\n");
fout(" {\n");
std::vector<SPGradientStop> stops = g->vector.stops;
if (stops.size() > 0)
{
fout(" stops:\n");
fout(" [\n");
for (unsigned int i = 0 ; i<stops.size() ; i++)
{
SPGradientStop stop = stops[i];
fout(" Stop\n");
fout(" {\n");
fout(" offset: %s\n", dstr(stop.offset).c_str());
fout(" color: %s\n", rgba(stop.color, stop.opacity).c_str());
fout(" },\n");
}
fout(" ]\n");
}
fout(" }\n");
fout("}\n");
fout("\n\n");
}
else if (SP_IS_RADIALGRADIENT(grad))
{
SPRadialGradient *g = SP_RADIALGRADIENT(grad);
fout("function %s() : RadialGradient\n", id.c_str());
fout("{\n");
fout(" return RadialGradient\n");
fout(" {\n");
fout(" centerX: %s\n", dstr(g->cx.value).c_str());
fout(" centerY: %s\n", dstr(g->cy.value).c_str());
fout(" focusX: %s\n", dstr(g->fx.value).c_str());
fout(" focusY: %s\n", dstr(g->fy.value).c_str());
fout(" radius: %s\n", dstr(g->r.value).c_str());
std::vector<SPGradientStop> stops = g->vector.stops;
if (stops.size() > 0)
{
fout(" stops:\n");
fout(" [\n");
for (unsigned int i = 0 ; i<stops.size() ; i++)
{
SPGradientStop stop = stops[i];
fout(" Stop\n");
fout(" {\n");
fout(" offset: %s\n", dstr(stop.offset).c_str());
fout(" color: %s\n", rgba(stop.color, stop.opacity).c_str());
fout(" },\n");
}
fout(" ]\n");
}
fout(" }\n");
fout("}\n");
fout("\n\n");
}
else
{
err("Unknown gradient type for '%s'\n", id.c_str());
return false;
}
return true;
}
/**
* Output an element's style attribute
*/
bool JavaFXOutput::doStyle(SPStyle *style)
{
if (!style)
return true;
out(" opacity: %s\n", dstr(effective_opacity(style)).c_str());
/**
* Fill
*/
SPIPaint fill = style->fill;
if (fill.isColor())
{
// see color.h for how to parse SPColor
out(" fill: %s\n",
rgba(fill.value.color, SP_SCALE24_TO_FLOAT(style->fill_opacity.value)).c_str());
}
else if (fill.isPaintserver())
{
if (fill.value.href && fill.value.href->getURI() )
{
String uri = fill.value.href->getURI()->toString();
if (uri.size()>0 && uri[0]=='#')
uri = uri.substr(1);
out(" fill: %s()\n", uri.c_str());
}
}
/**
* Stroke
*/
/**
*NOTE: Things in style we can use:
* SPIPaint stroke;
* SPILength stroke_width;
* SPIEnum stroke_linecap;
* SPIEnum stroke_linejoin;
* SPIFloat stroke_miterlimit;
* NRVpathDash stroke_dash;
* unsigned stroke_dasharray_set : 1;
* unsigned stroke_dasharray_inherit : 1;
* unsigned stroke_dashoffset_set : 1;
* SPIScale24 stroke_opacity;
*/
if (style->stroke_opacity.value > 0)
{
SPIPaint stroke = style->stroke;
out(" stroke: %s\n",
rgba(stroke.value.color, SP_SCALE24_TO_FLOAT(style->stroke_opacity.value)).c_str());
double strokewidth = style->stroke_width.value;
out(" strokeWidth: %s\n", dstr(strokewidth).c_str());
}
return true;
}
/**
* Output the curve data to buffer
*/
bool JavaFXOutput::doCurve(SPItem *item, const String &id)
{
using Geom::X;
using Geom::Y;
//### Get the Shape
if (!SP_IS_SHAPE(item))//Bulia's suggestion. Allow all shapes
return true;
SPShape *shape = SP_SHAPE(item);
SPCurve *curve = shape->curve;
if (curve->is_empty())
return true;
nrShapes++;
out(" SVGPath \n");
out(" {\n");
out(" id: \"%s\"\n", id.c_str());
/**
* Output the style information
*/
if (!doStyle(SP_OBJECT_STYLE(shape)))
return false;
// convert the path to only lineto's and cubic curveto's:
Geom::Scale yflip(1.0, -1.0);
Geom::Matrix tf = sp_item_i2d_affine(item) * yflip;
Geom::PathVector pathv = pathv_to_linear_and_cubic_beziers( curve->get_pathvector() * tf );
//Count the NR_CURVETOs/LINETOs (including closing line segment)
nrNodes = 0;
for(Geom::PathVector::const_iterator it = pathv.begin(); it != pathv.end(); ++it) {
nrNodes += (*it).size();
if (it->closed())
nrNodes += 1;
}
char *dataStr = sp_svg_write_path(pathv);
out(" content: \"%s\"\n", dataStr);
free(dataStr);
Geom::Rect cminmax( pathv.front().initialPoint(), pathv.front().initialPoint() );
/**
* Get the Min and Max X and Y extends for the Path.
* ....For all Subpaths in the <path>
*/
for (Geom::PathVector::const_iterator pit = pathv.begin(); pit != pathv.end(); ++pit)
{
cminmax.expandTo(pit->front().initialPoint());
/**
* For all segments in the subpath
*/
for (Geom::Path::const_iterator cit = pit->begin(); cit != pit->end_closed(); ++cit)
{
cminmax.expandTo(cit->finalPoint());
}
}
out(" },\n");
double cminx = cminmax.min()[X];
double cmaxx = cminmax.max()[X];
double cminy = cminmax.min()[Y];
double cmaxy = cminmax.max()[Y];
if (cminx < minx)
minx = cminx;
if (cmaxx > maxx)
maxx = cmaxx;
if (cminy < miny)
miny = cminy;
if (cmaxy > maxy)
maxy = cmaxy;
return true;
}
/**
* Output the tree data to buffer
*/
bool JavaFXOutput::doTreeRecursive(SPDocument *doc, SPObject *obj)
{
/**
* Check the type of node and process
*/
String id;
if (!obj->id)
{
char buf[16];
sprintf(buf, "id%d", idindex++);
id = buf;
}
else
{
id = obj->id;
}
if (SP_IS_ITEM(obj))
{
SPItem *item = SP_ITEM(obj);
if (!doCurve(item, id))
return false;
}
else if (SP_IS_GRADIENT(obj))
{
SPGradient *grad = SP_GRADIENT(obj);
if (!doGradient(grad, id))
return false;
}
/**
* Descend into children
*/
for (SPObject *child = obj->firstChild() ; child ; child = child->next)
{
if (!doTreeRecursive(doc, child))
return false;
}
return true;
}
/**
* Output the curve data to buffer
*/
bool JavaFXOutput::doTree(SPDocument *doc)
{
double bignum = 1000000.0;
minx = bignum;
maxx = -bignum;
miny = bignum;
maxy = -bignum;
if (!doTreeRecursive(doc, doc->root))
return false;
return true;
}
//########################################################################
//# M A I N O U T P U T
//########################################################################
/**
* Set values back to initial state
*/
void JavaFXOutput::reset()
{
nrNodes = 0;
nrShapes = 0;
idindex = 0;
name.clear();
outbuf.clear();
foutbuf.clear();
}
/**
* Saves the <paths> of an Inkscape SVG file as JavaFX spline definitions
*/
bool JavaFXOutput::saveDocument(SPDocument *doc, gchar const *uri)
{
reset();
name = Glib::path_get_basename(uri);
int pos = name.find('.');
if (pos > 0)
name = name.substr(0, pos);
//###### SAVE IN POV FORMAT TO BUFFER
//# Lets do the curves first, to get the stats
if (!doTree(doc))
return false;
String curveBuf = outbuf;
outbuf.clear();
if (!doHeader())
return false;
outbuf.append(curveBuf);
if (!doTail())
return false;
//###### WRITE TO FILE
FILE *f = Inkscape::IO::fopen_utf8name(uri, "w");
if (!f)
{
err("Could open JavaFX file '%s' for writing", uri);
return false;
}
for (String::iterator iter = outbuf.begin() ; iter!=outbuf.end(); iter++)
{
fputc(*iter, f);
}
fclose(f);
return true;
}
//########################################################################
//# EXTENSION API
//########################################################################
#include "clear-n_.h"
/**
* API call to save document
*/
void
JavaFXOutput::save(Inkscape::Extension::Output */*mod*/,
SPDocument *doc, gchar const *uri)
{
if (!saveDocument(doc, uri))
{
g_warning("Could not save JavaFX file '%s'", uri);
}
}
/**
* Make sure that we are in the database
*/
bool JavaFXOutput::check (Inkscape::Extension::Extension */*module*/)
{
/* We don't need a Key
if (NULL == Inkscape::Extension::db.get(SP_MODULE_KEY_OUTPUT_JFX))
return FALSE;
*/
return true;
}
/**
* This is the definition of JavaFX output. This function just
* calls the extension system with the memory allocated XML that
* describes the data.
*/
void
JavaFXOutput::init()
{
Inkscape::Extension::build_from_mem(
"<inkscape-extension xmlns=\"" INKSCAPE_EXTENSION_URI "\">\n"
"<name>" N_("JavaFX Output") "</name>\n"
"<id>org.inkscape.output.jfx</id>\n"
"<output>\n"
"<extension>.fx</extension>\n"
"<mimetype>text/x-javafx-script</mimetype>\n"
"<filetypename>" N_("JavaFX (*.fx)") "</filetypename>\n"
"<filetypetooltip>" N_("JavaFX Raytracer File") "</filetypetooltip>\n"
"</output>\n"
"</inkscape-extension>",
new JavaFXOutput());
}
} // 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:encoding=utf-8:textwidth=99 :