diff options
Diffstat (limited to 'src/svg/path-string.cpp')
| -rw-r--r-- | src/svg/path-string.cpp | 102 |
1 files changed, 100 insertions, 2 deletions
diff --git a/src/svg/path-string.cpp b/src/svg/path-string.cpp index 300650811..44b474dd2 100644 --- a/src/svg/path-string.cpp +++ b/src/svg/path-string.cpp @@ -13,7 +13,9 @@ */ #include "svg/path-string.h" +#include "svg/stringstream.h" #include "prefs-utils.h" +#include <algorithm> Inkscape::SVG::PathString::PathString() : allow_relative_coordinates(0 != prefs_get_int_attribute("options.svgoutput", "allowrelativecoordinates", 1)), @@ -25,7 +27,7 @@ void Inkscape::SVG::PathString::_appendOp(char abs_op, char rel_op) { bool rel_op_repeated = _rel_state.prevop == rel_op && !force_repeat_commands; unsigned int const abs_added_size = abs_op_repeated ? 0 : 2; unsigned int const rel_added_size = rel_op_repeated ? 0 : 2; - if ( _rel_state.str.size()+2 < _abs_state.str.size()+abs_added_size && allow_relative_coordinates ) { + if ( false && _rel_state.str.size()+2 < _abs_state.str.size()+abs_added_size && allow_relative_coordinates ) { // Copy rel to abs _abs_state = _rel_state; _abs_state.switches++; @@ -34,7 +36,7 @@ void Inkscape::SVG::PathString::_appendOp(char abs_op, char rel_op) { // _rel_state.str.size()+2 < _abs_state.str.size()+abs_added_size // _rel_state.str.size()+rel_added_size < _abs_state.str.size()+2 // _abs_state.str.size()+2 > _rel_state.str.size()+rel_added_size - } else if ( _abs_state.str.size()+2 < _rel_state.str.size()+rel_added_size ) { + } else if ( false && _abs_state.str.size()+2 < _rel_state.str.size()+rel_added_size ) { // Copy abs to rel _rel_state = _abs_state; _abs_state.switches++; @@ -44,6 +46,102 @@ void Inkscape::SVG::PathString::_appendOp(char abs_op, char rel_op) { if ( !rel_op_repeated ) _rel_state.appendOp(rel_op); } +void Inkscape::SVG::PathString::State::append(NR::Coord v) { + SVGOStringStream os; + os << ' ' << v; + str.append(os.str()); +} + +void Inkscape::SVG::PathString::State::append(NR::Point p) { + SVGOStringStream os; + os << ' ' << p[NR::X] << ',' << p[NR::Y]; + str.append(os.str()); +} + +void Inkscape::SVG::PathString::State::append(NR::Coord v, NR::Coord &rv) { + SVGOStringStream os; + os << ' ' << v; + str.append(os.str()); + double c; + sscanf(os.str().c_str(), " %lf", &c); + rv = c; +} + +void Inkscape::SVG::PathString::State::append(NR::Point p, NR::Point &rp) { + SVGOStringStream os; + os << ' ' << p[NR::X] << ',' << p[NR::Y]; + str.append(os.str()); + double x, y; + sscanf(os.str().c_str(), " %lf,%lf", &x, &y); + rp[NR::X] = x; + rp[NR::Y] = y; +} + +// NOTE: The following two appendRelative methods will not be exact if the total number of digits needed +// to represent the difference exceeds the precision of a double. This is not very likely though, and if +// it does happen the imprecise value is not likely to be chosen (because it will probably be a lot longer +// than the absolute value). + +void Inkscape::SVG::PathString::State::appendRelative(NR::Coord v, NR::Coord r) { + SVGOStringStream os; + int precision = (int)os.precision(); + int digitsBegin = (int)floor(log10(fabs(v-r))); // Position of first digit of difference + int digitsEnd = (int)floor(log10(std::min(fabs(v),fabs(r)))) - precision; // Position just beyond the last significant digit of the smallest (in absolute sense) number + os << ' '; + if (r == 0) { + os.precision(precision); + os << v; + } else if (v == 0) { + os.precision(precision); + os << -r; + } else if (digitsBegin>digitsEnd) { + os.precision(digitsBegin-digitsEnd); + os << (v-r); + } else { + // This assumes the input numbers are already rounded to 'precision' digits + os << '0'; + } + str.append(os.str()); +} + +void Inkscape::SVG::PathString::State::appendRelative(NR::Point p, NR::Point r) { + SVGOStringStream os; + int precision = (int)os.precision(); + int digitsBeginX = (int)floor(log10(fabs(p[NR::X]-r[NR::X]))); // Position of first digit of difference + int digitsEndX = (int)floor(log10(std::min(fabs(p[NR::X]),fabs(r[NR::X])))) - precision; // Position just beyond the last significant digit of the smallest (in absolute sense) number + int digitsBeginY = (int)floor(log10(fabs(p[NR::Y]-r[NR::Y]))); // Position of first digit of difference + int digitsEndY = (int)floor(log10(std::min(fabs(p[NR::Y]),fabs(r[NR::Y])))) - precision; // Position just beyond the last significant digit of the smallest (in absolute sense) number + os << ' '; + if (r[NR::X] == 0) { + os.precision(precision); + os << p[NR::X]; + } else if (p[NR::X] == 0) { + os.precision(precision); + os << -r[NR::X]; + } else if (digitsBeginX>digitsEndX) { + os.precision(digitsBeginX-digitsEndX); + os << (p[NR::X]-r[NR::X]); + } else { + // This assumes the input numbers are already rounded to 'precision' digits + os << '0'; + } + os << ','; + if (r[NR::Y] == 0) { + os.precision(precision); + os << p[NR::Y]; + } else if (p[NR::Y] == 0) { + os.precision(precision); + os << -r[NR::Y]; + } else if (digitsBeginY>digitsEndY) { + os.precision(digitsBeginY-digitsEndY); + os << (p[NR::Y]-r[NR::Y]); + } else { + // This assumes the input numbers are already rounded to 'precision' digits + os << '0'; + } + str.append(os.str()); +} + /* Local Variables: mode:c++ |
