b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof/*
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof * Released under GNU GPL, read the file 'COPYING' for more information
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof */
43a9ecef56abdf431763b9fb95469e70da237abbLiam P. White
43a9ecef56abdf431763b9fb95469e70da237abbLiam P. White#include <gtkmm.h>
43a9ecef56abdf431763b9fb95469e70da237abbLiam P. White
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "live_effects/lpe-simplify.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "display/curve.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "live_effects/parameter/parameter.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include <glibmm/i18n.h>
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "helper/geom.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "livarot/Path.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "splivarot.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include <2geom/svg-path-parser.h>
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "desktop.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "inkscape.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "svg/svg.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include "ui/tools/node-tool.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include <2geom/d2.h>
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include <2geom/generic-rect.h>
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof#include <2geom/interval.h>
a6b5d41707fe985d397907d52766cbcdcca9735fJabiertxof#include "ui/icon-names.h"
deaf8922dbca36e15fe43a72b2b4320ae649a42fJabiertxof#include "util/units.h"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofnamespace Inkscape {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofnamespace LivePathEffect {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::LPESimplify(LivePathEffectObject *lpeobject)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof : Effect(lpeobject),
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof steps(_("Steps:"),_("Change number of simplify steps "), "steps", &wr, this,1),
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof threshold(_("Roughly threshold:"), _("Roughly threshold:"), "threshold", &wr, this, 0.002),
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof smooth_angles(_("Smooth angles:"), _("Max degree difference on handles to perform a smooth"), "smooth_angles", &wr, this, 0.),
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof helper_size(_("Helper size:"), _("Helper size"), "helper_size", &wr, this, 5),
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof simplify_individual_paths(_("Paths separately"), _("Simplifying paths (separately)"), "simplify_individual_paths", &wr, this, false,
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof "", INKSCAPE_ICON("on"), INKSCAPE_ICON("off")),
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof simplify_just_coalesce(_("Just coalesce"), _("Simplify just coalesce"), "simplify_just_coalesce", &wr, this, false,
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof "", INKSCAPE_ICON("on"), INKSCAPE_ICON("off"))
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof{
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&steps);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&threshold);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&smooth_angles);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&helper_size);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&simplify_individual_paths);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof registerParameter(&simplify_just_coalesce);
26b55a439f4a219fec9c2b46a6b9d02640da6d76Jabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof threshold.param_set_range(0.0001, Geom::infinity());
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof threshold.param_set_increments(0.0001, 0.0001);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof threshold.param_set_digits(6);
26b55a439f4a219fec9c2b46a6b9d02640da6d76Jabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof steps.param_set_range(0, 100);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof steps.param_set_increments(1, 1);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof steps.param_set_digits(0);
26b55a439f4a219fec9c2b46a6b9d02640da6d76Jabiertxof
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof smooth_angles.param_set_range(0.0, 360.0);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof smooth_angles.param_set_increments(10, 10);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof smooth_angles.param_set_digits(2);
26b55a439f4a219fec9c2b46a6b9d02640da6d76Jabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof helper_size.param_set_range(0.0, 999.0);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof helper_size.param_set_increments(5, 5);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof helper_size.param_set_digits(2);
26b55a439f4a219fec9c2b46a6b9d02640da6d76Jabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof radius_helper_nodes = 6.0;
b8d3bf3a76982b925ccacab389e190adbfa1f3b1jabiertxof apply_to_clippath_and_mask = true;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::~LPESimplify() {}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofvoid
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::doBeforeEffect (SPLPEItem const* lpeitem)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(!hp.empty()) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp.clear();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof bbox = SP_ITEM(lpeitem)->visualBounds();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof radius_helper_nodes = helper_size;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofGtk::Widget *
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::newWidget()
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof // use manage here, because after deletion of Effect object, others might still be pointing to this widget.
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Gtk::VBox * vbox = Gtk::manage( new Gtk::VBox(Effect::newWidget()) );
98f9f27a27115988e05366a69d7b38be9c12f69dJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof vbox->set_border_width(5);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof vbox->set_homogeneous(false);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof vbox->set_spacing(2);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof std::vector<Parameter *>::iterator it = param_vector.begin();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Gtk::HBox * buttons = Gtk::manage(new Gtk::HBox(true,0));
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof while (it != param_vector.end()) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if ((*it)->widget_is_visible) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Parameter * param = *it;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Gtk::Widget * widg = dynamic_cast<Gtk::Widget *>(param->param_newWidget());
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if (param->param_key == "simplify_individual_paths" ||
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof param->param_key == "simplify_just_coalesce") {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Glib::ustring * tip = param->param_getTooltip();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if (widg) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof buttons->pack_start(*widg, true, true, 2);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if (tip) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_tooltip_text(*tip);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof } else {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_tooltip_text("");
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_has_tooltip(false);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof } else {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Glib::ustring * tip = param->param_getTooltip();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if (widg) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Gtk::HBox * horizontal_box = dynamic_cast<Gtk::HBox *>(widg);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof std::vector< Gtk::Widget* > child_list = horizontal_box->get_children();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Gtk::Entry* entry_widg = dynamic_cast<Gtk::Entry *>(child_list[1]);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof entry_widg->set_width_chars(8);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof vbox->pack_start(*widg, true, true, 2);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if (tip) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_tooltip_text(*tip);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof } else {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_tooltip_text("");
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof widg->set_has_tooltip(false);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof ++it;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof vbox->pack_start(*buttons,true, true, 2);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof return dynamic_cast<Gtk::Widget *>(vbox);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxofvoid
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2JabiertxofLPESimplify::doEffect(SPCurve *curve)
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof{
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Geom::PathVector const original_pathv = pathv_to_linear_and_cubic_beziers(curve->get_pathvector());
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof gdouble size = Geom::L2(bbox->dimensions());
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof //size /= Geom::Affine(0,0,0,0,0,0).descrim();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Path* pathliv = Path_for_pathvector(original_pathv);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(simplify_individual_paths) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof size = Geom::L2(Geom::bounds_fast(original_pathv)->dimensions());
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof size /= sp_lpe_item->i2doc_affine().descrim();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof for (int unsigned i = 0; i < steps; i++) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if ( simplify_just_coalesce ) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof pathliv->Coalesce(threshold * size);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof } else {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof pathliv->ConvertEvenLines(threshold * size);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof pathliv->Simplify(threshold * size);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof Geom::PathVector result = Geom::parse_svg_path(pathliv->svg_dump_path());
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof generateHelperPathAndSmooth(result);
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof curve->set_pathvector(result);
1bfcb8c7fbc1834069dbd15da41daf6cdfba6edfJabiertxof SPDesktop* desktop = SP_ACTIVE_DESKTOP;
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(desktop && INK_IS_NODE_TOOL(desktop->event_context)) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Inkscape::UI::Tools::NodeTool *nt = static_cast<Inkscape::UI::Tools::NodeTool*>(desktop->event_context);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof nt->update_helperpath();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofvoid
6103fefae34cb76c1ab70a8077454534aba342c5JabiertxofLPESimplify::generateHelperPathAndSmooth(Geom::PathVector &result)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(steps < 1) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof return;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::PathVector tmp_path;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Geom::CubicBezier const *cubic = NULL;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof for (Geom::PathVector::iterator path_it = result.begin(); path_it != result.end(); ++path_it) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if (path_it->empty()) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof continue;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński Geom::Path::iterator curve_it1 = path_it->begin(); // incoming curve
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński Geom::Path::iterator curve_it2 = ++(path_it->begin());// outgoing curve
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński Geom::Path::iterator curve_endit = path_it->end_default(); // this determines when the loop has to stop
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof SPCurve *nCurve = new SPCurve();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof if (path_it->closed()) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // if the path is closed, maybe we have to stop a bit earlier because the
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // closing line segment has zerolength.
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof const Geom::Curve &closingline =
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof path_it->back_closed(); // the closing line segment is always of type
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // Geom::LineSegment.
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if (are_near(closingline.initialPoint(), closingline.finalPoint())) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // closingline.isDegenerate() did not work, because it only checks for
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // *exact* zero length, which goes wrong for relative coordinates and
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // rounding errors...
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof // the closing line segment has zero-length. So stop before that one!
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof curve_endit = path_it->end_open();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(helper_size > 0) {
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof drawNode(curve_it1->initialPoint());
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof nCurve->moveto(curve_it1->initialPoint());
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof Geom::Point start = Geom::Point(0,0);
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof while (curve_it1 != curve_endit) {
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof cubic = dynamic_cast<Geom::CubicBezier const *>(&*curve_it1);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Point point_at1 = curve_it1->initialPoint();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Point point_at2 = curve_it1->finalPoint();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Point point_at3 = curve_it1->finalPoint();
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Point point_at4 = curve_it1->finalPoint();
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof if(start == Geom::Point(0,0)) {
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof start = point_at1;
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof }
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof if (cubic) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at1 = (*cubic)[1];
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at2 = (*cubic)[2];
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(path_it->closed() && curve_it2 == curve_endit) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at4 = start;
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(curve_it2 != curve_endit) {
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof cubic = dynamic_cast<Geom::CubicBezier const *>(&*curve_it2);
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof if (cubic) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at4 = (*cubic)[1];
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Ray ray1(point_at2, point_at3);
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof Geom::Ray ray2(point_at3, point_at4);
a16a494f042310ee849a6f717ffea70846f1f22cKrzysztof Kosiński double angle1 = Geom::deg_from_rad(ray1.angle());
a16a494f042310ee849a6f717ffea70846f1f22cKrzysztof Kosiński double angle2 = Geom::deg_from_rad(ray2.angle());
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof if((smooth_angles >= std::abs(angle2 - angle1)) && !are_near(point_at4,point_at3) && !are_near(point_at2,point_at3)) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof double dist = Geom::distance(point_at2,point_at3);
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof Geom::Angle angleFixed = ray2.angle();
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof angleFixed -= Geom::Angle::from_degrees(180.0);
006e2b15a435baa814a51ccfa7a729907d498ebdjabiertxof point_at2 = Geom::Point::polar(angleFixed, dist) + point_at3;
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof nCurve->curveto(point_at1, point_at2, curve_it1->finalPoint());
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof cubic = dynamic_cast<Geom::CubicBezier const *>(nCurve->last_segment());
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof if (cubic) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at1 = (*cubic)[1];
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof point_at2 = (*cubic)[2];
d7c3058cda5f2e4c0e7510579121e84ad7f2858aJabiertxof if(helper_size > 0) {
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(!are_near((*cubic)[0],(*cubic)[1])) {
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof drawHandle((*cubic)[1]);
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof drawHandleLine((*cubic)[0],(*cubic)[1]);
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(!are_near((*cubic)[3],(*cubic)[2])) {
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof drawHandle((*cubic)[2]);
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof drawHandleLine((*cubic)[3],(*cubic)[2]);
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof }
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof }
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof }
d7c3058cda5f2e4c0e7510579121e84ad7f2858aJabiertxof if(helper_size > 0) {
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof drawNode(curve_it1->finalPoint());
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof }
23e68a9a2e8c92bff1885ab56a74c13f6b4691a3Jabiertxof ++curve_it1;
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof ++curve_it2;
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof }
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof if (path_it->closed()) {
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof nCurve->closepath_current();
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof tmp_path.push_back(nCurve->get_pathvector()[0]);
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof nCurve->reset();
6103fefae34cb76c1ab70a8077454534aba342c5Jabiertxof delete nCurve;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof }
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof result = tmp_path;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxofvoid
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::drawNode(Geom::Point p)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof double r = radius_helper_nodes;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof char const * svgd;
1666b07512ebc5fa9136d5d9dd8e2a9c1416cc6eJabiertxof svgd = "M 0.55,0.5 A 0.05,0.05 0 0 1 0.5,0.55 0.05,0.05 0 0 1 0.45,0.5 0.05,0.05 0 0 1 0.5,0.45 0.05,0.05 0 0 1 0.55,0.5 Z M 0,0 1,0 1,1 0,1 Z";
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Geom::PathVector pathv = sp_svg_read_pathv(svgd);
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński pathv *= Geom::Scale(r) * Geom::Translate(p - Geom::Point(0.5*r,0.5*r));
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp.push_back(pathv[0]);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp.push_back(pathv[1]);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofvoid
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::drawHandle(Geom::Point p)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof double r = radius_helper_nodes;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof char const * svgd;
1666b07512ebc5fa9136d5d9dd8e2a9c1416cc6eJabiertxof svgd = "M 0.7,0.35 A 0.35,0.35 0 0 1 0.35,0.7 0.35,0.35 0 0 1 0,0.35 0.35,0.35 0 0 1 0.35,0 0.35,0.35 0 0 1 0.7,0.35 Z";
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Geom::PathVector pathv = sp_svg_read_pathv(svgd);
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński pathv *= Geom::Scale(r) * Geom::Translate(p - Geom::Point(0.35*r,0.35*r));
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp.push_back(pathv[0]);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofvoid
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::drawHandleLine(Geom::Point p,Geom::Point p2)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Geom::Path path;
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof path.start( p );
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof double diameter = radius_helper_nodes;
2ea303d5a80b7efba4d23ab33e7d5ea0c9b546b2Jabiertxof if(helper_size > 0 && Geom::distance(p,p2) > (diameter * 0.35)) {
64767c33508fddca523a1c9439f65aeaaf524d90Jabiertxof Geom::Ray ray2(p, p2);
64767c33508fddca523a1c9439f65aeaaf524d90Jabiertxof p2 = p2 - Geom::Point::polar(ray2.angle(),(diameter * 0.35));
64767c33508fddca523a1c9439f65aeaaf524d90Jabiertxof }
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof path.appendNew<Geom::LineSegment>( p2 );
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp.push_back(path);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofvoid
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxofLPESimplify::addCanvasIndicators(SPLPEItem const */*lpeitem*/, std::vector<Geom::PathVector> &hp_vec)
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof{
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof hp_vec.push_back(hp);
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}; //namespace LivePathEffect
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof}; /* namespace Inkscape */
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof/*
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof Local Variables:
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof mode:c++
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof c-file-style:"stroustrup"
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof indent-tabs-mode:nil
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof fill-column:99
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof End:
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof*/
b14e0c4cb620016f312432acc32db04b70f3fabcJabiertxof// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4 :