From 564ab243cc3fa7c0239d71a6dacb4a6c7765e9f5 Mon Sep 17 00:00:00 2001 From: jaseg <git@jaseg.de> Date: Thu, 3 Jun 2021 23:45:11 +0200 Subject: Add svg-flatten SVG feature tests --- svg-flatten/Makefile | 15 +- svg-flatten/src/nopencv_test.cpp | 405 --------------------- svg-flatten/src/test/nopencv_test.cpp | 405 +++++++++++++++++++++ svg-flatten/src/test/svg_tests.py | 75 ++++ svg-flatten/testdata/svg/circles.svg | 78 ++++ svg-flatten/testdata/svg/compound_xform.svg | 102 ++++++ svg-flatten/testdata/svg/empty.svg | 11 + svg-flatten/testdata/svg/empty_inkscape.svg | 54 +++ svg-flatten/testdata/svg/groups.svg | 96 +++++ svg-flatten/testdata/svg/pattern_fill.svg | 135 +++++++ svg-flatten/testdata/svg/pattern_stroke.svg | 89 +++++ svg-flatten/testdata/svg/pattern_stroke_dashed.svg | 89 +++++ svg-flatten/testdata/svg/rect.svg | 65 ++++ svg-flatten/testdata/svg/rect_occlusion.svg | 79 ++++ svg-flatten/testdata/svg/rotation.svg | 66 ++++ svg-flatten/testdata/svg/rotation_90.svg | 66 ++++ svg-flatten/testdata/svg/scale.svg | 91 +++++ svg-flatten/testdata/svg/shear.svg | 59 +++ svg-flatten/testdata/svg/stroke.svg | 67 ++++ svg-flatten/testdata/svg/stroke_caps.svg | 68 ++++ svg-flatten/testdata/svg/stroke_dashes.svg | 75 ++++ svg-flatten/testdata/svg/stroke_joins.svg | 68 ++++ svg-flatten/testdata/svg/text.svg | 108 ++++++ 23 files changed, 1952 insertions(+), 414 deletions(-) delete mode 100644 svg-flatten/src/nopencv_test.cpp create mode 100644 svg-flatten/src/test/nopencv_test.cpp create mode 100644 svg-flatten/src/test/svg_tests.py create mode 100644 svg-flatten/testdata/svg/circles.svg create mode 100644 svg-flatten/testdata/svg/compound_xform.svg create mode 100644 svg-flatten/testdata/svg/empty.svg create mode 100644 svg-flatten/testdata/svg/empty_inkscape.svg create mode 100644 svg-flatten/testdata/svg/groups.svg create mode 100644 svg-flatten/testdata/svg/pattern_fill.svg create mode 100644 svg-flatten/testdata/svg/pattern_stroke.svg create mode 100644 svg-flatten/testdata/svg/pattern_stroke_dashed.svg create mode 100644 svg-flatten/testdata/svg/rect.svg create mode 100644 svg-flatten/testdata/svg/rect_occlusion.svg create mode 100644 svg-flatten/testdata/svg/rotation.svg create mode 100644 svg-flatten/testdata/svg/rotation_90.svg create mode 100644 svg-flatten/testdata/svg/scale.svg create mode 100644 svg-flatten/testdata/svg/shear.svg create mode 100644 svg-flatten/testdata/svg/stroke.svg create mode 100644 svg-flatten/testdata/svg/stroke_caps.svg create mode 100644 svg-flatten/testdata/svg/stroke_dashes.svg create mode 100644 svg-flatten/testdata/svg/stroke_joins.svg create mode 100644 svg-flatten/testdata/svg/text.svg (limited to 'svg-flatten') diff --git a/svg-flatten/Makefile b/svg-flatten/Makefile index a980c05..2c9ff6c 100644 --- a/svg-flatten/Makefile +++ b/svg-flatten/Makefile @@ -1,5 +1,5 @@ -CXX := clang++ +CXX ?= clang++ LD ?= ld INSTALL := install PKG_CONFIG ?= pkg-config @@ -45,7 +45,7 @@ SOURCES += $(CLIPPER_SOURCES) INCLUDES := -Iinclude -Isrc $(CLIPPER_INCLUDES) $(VORONOI_INCLUDES) $(POISSON_INCLUDES) $(BASE64_INCLUDES) $(ARGAGG_INCLUDES) $(CAVC_INCLUDES) $(SUBPROCESS_INCLUDES) $(MINUNIT_INCLUDES) $(STB_INCLUDES) CXXFLAGS := -std=c++2a -g -Wall -Wextra -O0 -LDFLAGS := -lm -lc -lstdc++ +LDFLAGS := -lm PKG_CONFIG_DEPS := ifdef USE_SYSTEM_PUGIXML @@ -71,19 +71,16 @@ $(BUILDDIR)/%.o: %.cpp $(BUILDDIR)/$(TARGET): $(SOURCES:%.cpp=$(BUILDDIR)/%.o) @mkdir -p $(dir $@) - if [ $$(uname -s) = "Darwin" ]; then \ - $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $^; \ - else \ - $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ -Wl,--start-group $^ -lstdc++fs -Wl,--end-group; \ - fi + $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $^ -$(BUILDDIR)/nopencv-tests: src/nopencv_test.cpp src/nopencv.cpp +$(BUILDDIR)/nopencv-test: src/test/nopencv_test.cpp src/nopencv.cpp @mkdir -p $(dir $@) $(CXX) $(CXXFLAGS) $(INCLUDES) $(LDFLAGS) -o $@ $^ .PHONY: tests -tests: $(BUILDDIR)/nopencv-tests +tests: $(BUILDDIR)/nopencv-test + $(BUILDDIR)/nopencv-test .PHONY: install install: diff --git a/svg-flatten/src/nopencv_test.cpp b/svg-flatten/src/nopencv_test.cpp deleted file mode 100644 index 17ba71e..0000000 --- a/svg-flatten/src/nopencv_test.cpp +++ /dev/null @@ -1,405 +0,0 @@ - -#include <iostream> -#include <fstream> -#include <iomanip> -#include <cmath> -#include <filesystem> - -#include "nopencv.hpp" - -#include <subprocess.h> -#include <minunit.h> - -#include "stb_image.h" - -using namespace gerbolyze; -using namespace gerbolyze::nopencv; - -char msg[1024]; - -class TempfileHack { -public: - TempfileHack(const string ext) : m_path { filesystem::temp_directory_path() / (std::tmpnam(nullptr) + ext) } {} - ~TempfileHack() { remove(m_path); } - - const char *c_str() { return m_path.c_str(); } - -private: - filesystem::path m_path; -}; - -class SVGPolyRenderer { -public: - SVGPolyRenderer(const char *fn, int width_px, int height_px) - : m_svg(fn) { - m_svg << "<svg width=\"" << width_px << "px\" height=\"" << height_px << "px\" viewBox=\"0 0 " - << width_px << " " << height_px << "\" " - << "xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">" << endl; - m_svg << "<rect width=\"100%\" height=\"100%\" fill=\"black\"/>" << endl; - } - - ContourCallback callback() { - return [this](Polygon_i &poly, ContourPolarity pol) { - mu_assert(poly.size() > 0, "Empty contour returned"); - mu_assert(poly.size() > 2, "Contour has less than three points, no area"); - mu_assert(pol == CP_CONTOUR || pol == CP_HOLE, "Contour has invalid polarity"); - - m_svg << "<path fill=\"" << ((pol == CP_HOLE) ? "black" : "white") << "\" d=\""; - m_svg << "M " << poly[0][0] << " " << poly[0][1]; - for (size_t i=1; i<poly.size(); i++) { - m_svg << " L " << poly[i][0] << " " << poly[i][1]; - } - m_svg << " Z\"/>" << endl; - }; - } - - void close() { - m_svg << "</svg>" << endl; - m_svg.close(); - } - -private: - ofstream m_svg; -}; - -MU_TEST(test_complex_example_from_paper) { - int32_t img_data[6*9] = { - 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 1, 1, 1, 1, 1, 1, 1, 0, - 0, 1, 0, 0, 1, 0, 0, 1, 0, - 0, 1, 0, 0, 1, 0, 0, 1, 0, - 0, 1, 1, 1, 1, 1, 1, 1, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, - }; - Image32 test_img(9, 6, static_cast<int*>(img_data)); - - const Polygon_i expected_polys[3] = { - { - {1,1}, {1,2}, {1,3}, {1,4}, {1,5}, - {2,5}, {3,5}, {4,5}, {5,5}, {6,5}, {7,5}, {8,5}, - {8,4}, {8,3}, {8,2}, {8,1}, - {7,1}, {6,1}, {5,1}, {4,1}, {3,1}, {2,1} - }, - { - {2,2}, {2,3}, {2,4}, - {3,4}, {4,4}, - {4,3}, {4,2}, - {3,2} - }, - { - {5,2}, {5,3}, {5,4}, - {6,4}, {7,4}, - {7,3}, {7,2}, - {6,2} - } - }; - - const ContourPolarity expected_polarities[3] = {CP_CONTOUR, CP_HOLE, CP_HOLE}; - - int invocation_count = 0; - gerbolyze::nopencv::find_contours(test_img, [&invocation_count, &expected_polarities, &expected_polys](Polygon_i &poly, ContourPolarity pol) { - invocation_count += 1; - mu_assert((invocation_count <= 3), "Too many contours returned"); - - mu_assert(poly.size() > 0, "Empty contour returned"); - mu_assert_int_eq(pol, expected_polarities[invocation_count-1]); - - i2p last; - bool first = true; - Polygon_i exp = expected_polys[invocation_count-1]; - for (auto &p : poly) { - if (!first) { - mu_assert((fabs(p[0] - last[0]) + fabs(p[1] - last[1]) == 1), "Subsequent contour points have distance other than one"); - mu_assert(find(exp.begin(), exp.end(), p) != exp.end(), "Got unexpected contour point"); - } - last = p; - } - }); - mu_assert_int_eq(3, invocation_count); - - int32_t tpl[6*9] = { - 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 2, 2, 2, 2, 2, 2,-2, 0, - 0,-3, 0, 0,-4, 0, 0,-2, 0, - 0,-3, 0, 0,-4, 0, 0,-2, 0, - 0, 2, 2, 2, 2, 2, 2,-2, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, - }; - - - for (int y=0; y<6; y++) { - for (int x=0; x<9; x++) { - int a = test_img.at(x, y), b = tpl[y*9+x]; - if (a != b) { - cout << "Result:" << endl; - cout << " "; - for (int x=0; x<9; x++) { - cout << x << " "; - } - cout << endl; - cout << " "; - for (int x=0; x<9; x++) { - cout << "---"; - } - cout << endl; - for (int y=0; y<6; y++) { - cout << y << " | "; - for (int x=0; x<9; x++) { - cout << setfill(' ') << setw(2) << test_img.at(x, y) << " "; - } - cout << endl; - } - - snprintf(msg, sizeof(msg), "Result does not match template @(%d, %d): %d != %d\n", x, y, a, b); - mu_fail(msg); - } - } - } -} - -int render_svg(const char *in_svg, const char *out_png) { - const char *command_line[] = {"resvg", in_svg, out_png, nullptr}; - struct subprocess_s subprocess; - int rc = subprocess_create(command_line, subprocess_option_inherit_environment, &subprocess); - if (rc) - return rc; - - int resvg_rc = -1; - rc = subprocess_join(&subprocess, &resvg_rc); - if (rc) - return rc; - if (resvg_rc) - return -resvg_rc; - - rc = subprocess_destroy(&subprocess); - if (rc) - return rc; - - return 0; -} - -static void testdata_roundtrip(const char *fn) { - Image32 ref_img; - mu_assert(ref_img.load(fn), "Input image failed to load"); - ref_img.binarize(); - Image32 ref_img_copy(ref_img); - - TempfileHack tmp_svg(".svg"); - TempfileHack tmp_png(".png"); - - SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows()); - gerbolyze::nopencv::find_contours(ref_img, ctx.callback()); - ctx.close(); - - mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str())); - - Image32 out_img; - mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load"); - out_img.binarize(); - - mu_assert_int_eq(ref_img.cols(), out_img.cols()); - mu_assert_int_eq(ref_img.rows(), out_img.rows()); - - for (int y=0; y<out_img.rows(); y++) { - for (int x=0; x<out_img.cols(); x++) { - if (out_img.at(x, y) != ref_img_copy.at(x, y)) { - snprintf(msg, sizeof(msg), "%s: Result does not match input @(%d, %d): %d != %d\n", fn, x, y, out_img.at(x, y), ref_img_copy.at(x, y)); - mu_fail(msg); - } - } - } -} - -MU_TEST(test_round_trip_blank) { testdata_roundtrip("testdata/blank.png"); } -MU_TEST(test_round_trip_white) { testdata_roundtrip("testdata/white.png"); } -MU_TEST(test_round_trip_blob_border_w) { testdata_roundtrip("testdata/blob-border-w.png"); } -MU_TEST(test_round_trip_blobs_borders) { testdata_roundtrip("testdata/blobs-borders.png"); } -MU_TEST(test_round_trip_blobs_corners) { testdata_roundtrip("testdata/blobs-corners.png"); } -MU_TEST(test_round_trip_blobs_crossing) { testdata_roundtrip("testdata/blobs-crossing.png"); } -MU_TEST(test_round_trip_cross) { testdata_roundtrip("testdata/cross.png"); } -MU_TEST(test_round_trip_letter_e) { testdata_roundtrip("testdata/letter-e.png"); } -MU_TEST(test_round_trip_paper_example) { testdata_roundtrip("testdata/paper-example.png"); } -MU_TEST(test_round_trip_paper_example_inv) { testdata_roundtrip("testdata/paper-example-inv.png"); } -MU_TEST(test_round_trip_single_px) { testdata_roundtrip("testdata/single-px.png"); } -MU_TEST(test_round_trip_single_px_inv) { testdata_roundtrip("testdata/single-px-inv.png"); } -MU_TEST(test_round_trip_two_blobs) { testdata_roundtrip("testdata/two-blobs.png"); } -MU_TEST(test_round_trip_two_px) { testdata_roundtrip("testdata/two-px.png"); } -MU_TEST(test_round_trip_two_px_inv) { testdata_roundtrip("testdata/two-px-inv.png"); } - -static void test_polygon_area(const char *fn) { - //cerr << endl << "poly area test " << fn << endl; - Image32 ref_img; - mu_assert(ref_img.load(fn), "Input image failed to load"); - ref_img.binarize(); - - int white_px_count = 0; - int black_px_count = 0; - for (int y=0; y<ref_img.rows(); y++) { - for (int x=0; x<ref_img.cols(); x++) { - if (ref_img.at(x, y)) { - white_px_count += 1; - } else { - black_px_count += 1; - } - } - } - - double pos_sum = 0.0; - double neg_sum = ref_img.size(); - gerbolyze::nopencv::find_contours(ref_img, [&pos_sum, &neg_sum](Polygon_i& poly, ContourPolarity pol) { - double area = polygon_area(poly); - //cerr << endl << fn << ": " << area << " " << pos_sum << " / " << neg_sum << " -- " << white_px_count << " / " << black_px_count << " GOT: " << poly.size() << " w/ " << pol << endl; - mu_assert(fabs(area) > 0.99, "Polygon smaller than a single pixel"); - mu_assert((pol == CP_CONTOUR) == (area >= 0), "Polygon area has incorrect sign"); - - pos_sum += area; - neg_sum -= area; - }); - - mu_assert(pos_sum - white_px_count < 0.01, "Calculated area outside tolerance"); - mu_assert(neg_sum - black_px_count < 0.01, "Calculated area outside tolerance"); - //cerr << endl << "poly area test " << fn << " done" << endl; -} - -MU_TEST(test_polygon_area_blank) { test_polygon_area("testdata/blank.png"); } -MU_TEST(test_polygon_area_white) { test_polygon_area("testdata/white.png"); } -MU_TEST(test_polygon_area_blob_border_w) { test_polygon_area("testdata/blob-border-w.png"); } -MU_TEST(test_polygon_area_blobs_borders) { test_polygon_area("testdata/blobs-borders.png"); } -MU_TEST(test_polygon_area_blobs_corners) { test_polygon_area("testdata/blobs-corners.png"); } -MU_TEST(test_polygon_area_blobs_crossing) { test_polygon_area("testdata/blobs-crossing.png"); } -MU_TEST(test_polygon_area_cross) { test_polygon_area("testdata/cross.png"); } -MU_TEST(test_polygon_area_letter_e) { test_polygon_area("testdata/letter-e.png"); } -MU_TEST(test_polygon_area_paper_example) { test_polygon_area("testdata/paper-example.png"); } -MU_TEST(test_polygon_area_paper_example_inv) { test_polygon_area("testdata/paper-example-inv.png"); } -MU_TEST(test_polygon_area_single_px) { test_polygon_area("testdata/single-px.png"); } -MU_TEST(test_polygon_area_single_px_inv) { test_polygon_area("testdata/single-px-inv.png"); } -MU_TEST(test_polygon_area_two_blobs) { test_polygon_area("testdata/two-blobs.png"); } -MU_TEST(test_polygon_area_two_px) { test_polygon_area("testdata/two-px.png"); } -MU_TEST(test_polygon_area_two_px_inv) { test_polygon_area("testdata/two-px-inv.png"); } - -static void chain_approx_test(const char *fn) { - //cout << endl << "Testing \"" << fn << "\"" << endl; - Image32 ref_img; - mu_assert(ref_img.load(fn), "Input image failed to load"); - ref_img.binarize(); - Image32 ref_img_copy(ref_img); - - TempfileHack tmp_svg(".svg"); - TempfileHack tmp_png(".png"); - - SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows()); - gerbolyze::nopencv::find_contours(ref_img, simplify_contours_teh_chin(ctx.callback())); - ctx.close(); - - mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str())); - - Image32 out_img; - mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load"); - mu_assert_int_eq(ref_img.rows(), out_img.rows()); - mu_assert_int_eq(ref_img.cols(), out_img.cols()); - - double max_abs_deviation = 0; - double rms_sum = 0; - double mean_sum = 0; - for (int y=0; y<out_img.rows(); y++) { - for (int x=0; x<out_img.cols(); x++) { - double delta = fabs((double)out_img.at(x, y)/255.0 - (double)ref_img_copy.at(x, y)); - max_abs_deviation = fmax(max_abs_deviation, delta); - rms_sum += delta*delta; - mean_sum += delta; - } - } - - rms_sum = sqrt(rms_sum / out_img.size()); - mean_sum /= out_img.size(); - if (rms_sum > 0.5) { - snprintf(msg, sizeof(msg), "%s: Chain approximation RMS error is above threshold: %.3f > 0.5\n", fn, rms_sum); - mu_fail(msg); - } - if (mean_sum > 0.1) { - snprintf(msg, sizeof(msg), "%s: Chain approximation mean error is above threshold: %.3f > 0.1\n", fn, mean_sum); - mu_fail(msg); - } - //mu_assert(max_abs_deviation < 0.5, "Maximum chain approximation error is above threshold"); -} - - -MU_TEST(chain_approx_test_chromosome) { chain_approx_test("testdata/chain-approx-teh-chin-chromosome.png"); } -MU_TEST(chain_approx_test_blank) { chain_approx_test("testdata/blank.png"); } -MU_TEST(chain_approx_test_white) { chain_approx_test("testdata/white.png"); } -MU_TEST(chain_approx_test_blob_border_w) { chain_approx_test("testdata/blob-border-w.png"); } -MU_TEST(chain_approx_test_blobs_borders) { chain_approx_test("testdata/blobs-borders.png"); } -MU_TEST(chain_approx_test_blobs_corners) { chain_approx_test("testdata/blobs-corners.png"); } -MU_TEST(chain_approx_test_blobs_crossing) { chain_approx_test("testdata/blobs-crossing.png"); } -MU_TEST(chain_approx_test_cross) { chain_approx_test("testdata/cross.png"); } -MU_TEST(chain_approx_test_letter_e) { chain_approx_test("testdata/letter-e.png"); } -MU_TEST(chain_approx_test_paper_example) { chain_approx_test("testdata/paper-example.png"); } -MU_TEST(chain_approx_test_paper_example_inv) { chain_approx_test("testdata/paper-example-inv.png"); } -MU_TEST(chain_approx_test_single_px) { chain_approx_test("testdata/single-px.png"); } -MU_TEST(chain_approx_test_single_px_inv) { chain_approx_test("testdata/single-px-inv.png"); } -MU_TEST(chain_approx_test_two_blobs) { chain_approx_test("testdata/two-blobs.png"); } -MU_TEST(chain_approx_test_two_px) { chain_approx_test("testdata/two-px.png"); } -MU_TEST(chain_approx_test_two_px_inv) { chain_approx_test("testdata/two-px-inv.png"); } - - -MU_TEST_SUITE(nopencv_contours_suite) { - MU_RUN_TEST(test_complex_example_from_paper); - - MU_RUN_TEST(test_round_trip_blank); - MU_RUN_TEST(test_round_trip_white); - MU_RUN_TEST(test_round_trip_blob_border_w); - MU_RUN_TEST(test_round_trip_blobs_borders); - MU_RUN_TEST(test_round_trip_blobs_corners); - MU_RUN_TEST(test_round_trip_blobs_crossing); - MU_RUN_TEST(test_round_trip_cross); - MU_RUN_TEST(test_round_trip_letter_e); - MU_RUN_TEST(test_round_trip_paper_example); - MU_RUN_TEST(test_round_trip_paper_example_inv); - MU_RUN_TEST(test_round_trip_single_px); - MU_RUN_TEST(test_round_trip_single_px_inv); - MU_RUN_TEST(test_round_trip_two_blobs); - MU_RUN_TEST(test_round_trip_two_px); - MU_RUN_TEST(test_round_trip_two_px_inv); - - MU_RUN_TEST(chain_approx_test_chromosome); - MU_RUN_TEST(chain_approx_test_blank); - MU_RUN_TEST(chain_approx_test_white); - MU_RUN_TEST(chain_approx_test_blob_border_w); - MU_RUN_TEST(chain_approx_test_blobs_borders); - MU_RUN_TEST(chain_approx_test_blobs_corners); - MU_RUN_TEST(chain_approx_test_blobs_crossing); - MU_RUN_TEST(chain_approx_test_cross); - MU_RUN_TEST(chain_approx_test_letter_e); - MU_RUN_TEST(chain_approx_test_paper_example); - MU_RUN_TEST(chain_approx_test_paper_example_inv); - MU_RUN_TEST(chain_approx_test_single_px); - MU_RUN_TEST(chain_approx_test_single_px_inv); - MU_RUN_TEST(chain_approx_test_two_blobs); - MU_RUN_TEST(chain_approx_test_two_px); - MU_RUN_TEST(chain_approx_test_two_px_inv); - - MU_RUN_TEST(test_polygon_area_blank); - MU_RUN_TEST(test_polygon_area_white); - MU_RUN_TEST(test_polygon_area_blob_border_w); - MU_RUN_TEST(test_polygon_area_blobs_borders); - MU_RUN_TEST(test_polygon_area_blobs_corners); - MU_RUN_TEST(test_polygon_area_blobs_crossing); - MU_RUN_TEST(test_polygon_area_cross); - MU_RUN_TEST(test_polygon_area_letter_e); - MU_RUN_TEST(test_polygon_area_paper_example); - MU_RUN_TEST(test_polygon_area_paper_example_inv); - MU_RUN_TEST(test_polygon_area_single_px); - MU_RUN_TEST(test_polygon_area_single_px_inv); - MU_RUN_TEST(test_polygon_area_two_blobs); - MU_RUN_TEST(test_polygon_area_two_px); - MU_RUN_TEST(test_polygon_area_two_px_inv); -}; - -int main(int argc, char **argv) { - (void)argc; - (void)argv; - - MU_RUN_SUITE(nopencv_contours_suite); - MU_REPORT(); - return MU_EXIT_CODE; -} diff --git a/svg-flatten/src/test/nopencv_test.cpp b/svg-flatten/src/test/nopencv_test.cpp new file mode 100644 index 0000000..2f358b2 --- /dev/null +++ b/svg-flatten/src/test/nopencv_test.cpp @@ -0,0 +1,405 @@ + +#include <iostream> +#include <fstream> +#include <iomanip> +#include <cmath> +#include <filesystem> + +#include "nopencv.hpp" + +#include <subprocess.h> +#include <minunit.h> + +#include "stb_image.h" + +using namespace gerbolyze; +using namespace gerbolyze::nopencv; + +char msg[512]; + +class TempfileHack { +public: + TempfileHack(const string ext) : m_path { filesystem::temp_directory_path() / (std::tmpnam(nullptr) + ext) } {} + ~TempfileHack() { remove(m_path); } + + const char *c_str() { return m_path.c_str(); } + +private: + filesystem::path m_path; +}; + +class SVGPolyRenderer { +public: + SVGPolyRenderer(const char *fn, int width_px, int height_px) + : m_svg(fn) { + m_svg << "<svg width=\"" << width_px << "px\" height=\"" << height_px << "px\" viewBox=\"0 0 " + << width_px << " " << height_px << "\" " + << "xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">" << endl; + m_svg << "<rect width=\"100%\" height=\"100%\" fill=\"black\"/>" << endl; + } + + ContourCallback callback() { + return [this](Polygon_i &poly, ContourPolarity pol) { + mu_assert(poly.size() > 0, "Empty contour returned"); + mu_assert(poly.size() > 2, "Contour has less than three points, no area"); + mu_assert(pol == CP_CONTOUR || pol == CP_HOLE, "Contour has invalid polarity"); + + m_svg << "<path fill=\"" << ((pol == CP_HOLE) ? "black" : "white") << "\" d=\""; + m_svg << "M " << poly[0][0] << " " << poly[0][1]; + for (size_t i=1; i<poly.size(); i++) { + m_svg << " L " << poly[i][0] << " " << poly[i][1]; + } + m_svg << " Z\"/>" << endl; + }; + } + + void close() { + m_svg << "</svg>" << endl; + m_svg.close(); + } + +private: + ofstream m_svg; +}; + +MU_TEST(test_complex_example_from_paper) { + int32_t img_data[6*9] = { + 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 1, 1, 1, 1, 1, 1, 1, 0, + 0, 1, 0, 0, 1, 0, 0, 1, 0, + 0, 1, 0, 0, 1, 0, 0, 1, 0, + 0, 1, 1, 1, 1, 1, 1, 1, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, + }; + Image32 test_img(9, 6, static_cast<int*>(img_data)); + + const Polygon_i expected_polys[3] = { + { + {1,1}, {1,2}, {1,3}, {1,4}, {1,5}, + {2,5}, {3,5}, {4,5}, {5,5}, {6,5}, {7,5}, {8,5}, + {8,4}, {8,3}, {8,2}, {8,1}, + {7,1}, {6,1}, {5,1}, {4,1}, {3,1}, {2,1} + }, + { + {2,2}, {2,3}, {2,4}, + {3,4}, {4,4}, + {4,3}, {4,2}, + {3,2} + }, + { + {5,2}, {5,3}, {5,4}, + {6,4}, {7,4}, + {7,3}, {7,2}, + {6,2} + } + }; + + const ContourPolarity expected_polarities[3] = {CP_CONTOUR, CP_HOLE, CP_HOLE}; + + int invocation_count = 0; + gerbolyze::nopencv::find_contours(test_img, [&invocation_count, &expected_polarities, &expected_polys](Polygon_i &poly, ContourPolarity pol) { + invocation_count += 1; + mu_assert((invocation_count <= 3), "Too many contours returned"); + + mu_assert(poly.size() > 0, "Empty contour returned"); + mu_assert_int_eq(pol, expected_polarities[invocation_count-1]); + + i2p last; + bool first = true; + Polygon_i exp = expected_polys[invocation_count-1]; + for (auto &p : poly) { + if (!first) { + mu_assert((fabs(p[0] - last[0]) + fabs(p[1] - last[1]) == 1), "Subsequent contour points have distance other than one"); + mu_assert(find(exp.begin(), exp.end(), p) != exp.end(), "Got unexpected contour point"); + } + last = p; + } + }); + mu_assert_int_eq(3, invocation_count); + + int32_t tpl[6*9] = { + 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 2, 2, 2, 2, 2, 2,-2, 0, + 0,-3, 0, 0,-4, 0, 0,-2, 0, + 0,-3, 0, 0,-4, 0, 0,-2, 0, + 0, 2, 2, 2, 2, 2, 2,-2, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, + }; + + + for (int y=0; y<6; y++) { + for (int x=0; x<9; x++) { + int a = test_img.at(x, y), b = tpl[y*9+x]; + if (a != b) { + cout << "Result:" << endl; + cout << " "; + for (int x=0; x<9; x++) { + cout << x << " "; + } + cout << endl; + cout << " "; + for (int x=0; x<9; x++) { + cout << "---"; + } + cout << endl; + for (int y=0; y<6; y++) { + cout << y << " | "; + for (int x=0; x<9; x++) { + cout << setfill(' ') << setw(2) << test_img.at(x, y) << " "; + } + cout << endl; + } + + snprintf(msg, sizeof(msg), "Result does not match template @(%d, %d): %d != %d\n", x, y, a, b); + mu_fail(msg); + } + } + } +} + +int render_svg(const char *in_svg, const char *out_png) { + const char *command_line[] = {"resvg", in_svg, out_png, nullptr}; + struct subprocess_s subprocess; + int rc = subprocess_create(command_line, subprocess_option_inherit_environment, &subprocess); + if (rc) + return rc; + + int resvg_rc = -1; + rc = subprocess_join(&subprocess, &resvg_rc); + if (rc) + return rc; + if (resvg_rc) + return -resvg_rc; + + rc = subprocess_destroy(&subprocess); + if (rc) + return rc; + + return 0; +} + +static void testdata_roundtrip(const char *fn) { + Image32 ref_img; + mu_assert(ref_img.load(fn), "Input image failed to load"); + ref_img.binarize(); + Image32 ref_img_copy(ref_img); + + TempfileHack tmp_svg(".svg"); + TempfileHack tmp_png(".png"); + + SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows()); + gerbolyze::nopencv::find_contours(ref_img, ctx.callback()); + ctx.close(); + + mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str())); + + Image32 out_img; + mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load"); + out_img.binarize(); + + mu_assert_int_eq(ref_img.cols(), out_img.cols()); + mu_assert_int_eq(ref_img.rows(), out_img.rows()); + + for (int y=0; y<out_img.rows(); y++) { + for (int x=0; x<out_img.cols(); x++) { + if (out_img.at(x, y) != ref_img_copy.at(x, y)) { + snprintf(msg, sizeof(msg), "%s: Result does not match input @(%d, %d): %d != %d\n", fn, x, y, out_img.at(x, y), ref_img_copy.at(x, y)); + mu_fail(msg); + } + } + } +} + +MU_TEST(test_round_trip_blank) { testdata_roundtrip("testdata/blank.png"); } +MU_TEST(test_round_trip_white) { testdata_roundtrip("testdata/white.png"); } +MU_TEST(test_round_trip_blob_border_w) { testdata_roundtrip("testdata/blob-border-w.png"); } +MU_TEST(test_round_trip_blobs_borders) { testdata_roundtrip("testdata/blobs-borders.png"); } +MU_TEST(test_round_trip_blobs_corners) { testdata_roundtrip("testdata/blobs-corners.png"); } +MU_TEST(test_round_trip_blobs_crossing) { testdata_roundtrip("testdata/blobs-crossing.png"); } +MU_TEST(test_round_trip_cross) { testdata_roundtrip("testdata/cross.png"); } +MU_TEST(test_round_trip_letter_e) { testdata_roundtrip("testdata/letter-e.png"); } +MU_TEST(test_round_trip_paper_example) { testdata_roundtrip("testdata/paper-example.png"); } +MU_TEST(test_round_trip_paper_example_inv) { testdata_roundtrip("testdata/paper-example-inv.png"); } +MU_TEST(test_round_trip_single_px) { testdata_roundtrip("testdata/single-px.png"); } +MU_TEST(test_round_trip_single_px_inv) { testdata_roundtrip("testdata/single-px-inv.png"); } +MU_TEST(test_round_trip_two_blobs) { testdata_roundtrip("testdata/two-blobs.png"); } +MU_TEST(test_round_trip_two_px) { testdata_roundtrip("testdata/two-px.png"); } +MU_TEST(test_round_trip_two_px_inv) { testdata_roundtrip("testdata/two-px-inv.png"); } + +static void test_polygon_area(const char *fn) { + //cerr << endl << "poly area test " << fn << endl; + Image32 ref_img; + mu_assert(ref_img.load(fn), "Input image failed to load"); + ref_img.binarize(); + + int white_px_count = 0; + int black_px_count = 0; + for (int y=0; y<ref_img.rows(); y++) { + for (int x=0; x<ref_img.cols(); x++) { + if (ref_img.at(x, y)) { + white_px_count += 1; + } else { + black_px_count += 1; + } + } + } + + double pos_sum = 0.0; + double neg_sum = ref_img.size(); + gerbolyze::nopencv::find_contours(ref_img, [&pos_sum, &neg_sum](Polygon_i& poly, ContourPolarity pol) { + double area = polygon_area(poly); + //cerr << endl << fn << ": " << area << " " << pos_sum << " / " << neg_sum << " -- " << white_px_count << " / " << black_px_count << " GOT: " << poly.size() << " w/ " << pol << endl; + mu_assert(fabs(area) > 0.99, "Polygon smaller than a single pixel"); + mu_assert((pol == CP_CONTOUR) == (area >= 0), "Polygon area has incorrect sign"); + + pos_sum += area; + neg_sum -= area; + }); + + mu_assert(pos_sum - white_px_count < 0.01, "Calculated area outside tolerance"); + mu_assert(neg_sum - black_px_count < 0.01, "Calculated area outside tolerance"); + //cerr << endl << "poly area test " << fn << " done" << endl; +} + +MU_TEST(test_polygon_area_blank) { test_polygon_area("testdata/blank.png"); } +MU_TEST(test_polygon_area_white) { test_polygon_area("testdata/white.png"); } +MU_TEST(test_polygon_area_blob_border_w) { test_polygon_area("testdata/blob-border-w.png"); } +MU_TEST(test_polygon_area_blobs_borders) { test_polygon_area("testdata/blobs-borders.png"); } +MU_TEST(test_polygon_area_blobs_corners) { test_polygon_area("testdata/blobs-corners.png"); } +MU_TEST(test_polygon_area_blobs_crossing) { test_polygon_area("testdata/blobs-crossing.png"); } +MU_TEST(test_polygon_area_cross) { test_polygon_area("testdata/cross.png"); } +MU_TEST(test_polygon_area_letter_e) { test_polygon_area("testdata/letter-e.png"); } +MU_TEST(test_polygon_area_paper_example) { test_polygon_area("testdata/paper-example.png"); } +MU_TEST(test_polygon_area_paper_example_inv) { test_polygon_area("testdata/paper-example-inv.png"); } +MU_TEST(test_polygon_area_single_px) { test_polygon_area("testdata/single-px.png"); } +MU_TEST(test_polygon_area_single_px_inv) { test_polygon_area("testdata/single-px-inv.png"); } +MU_TEST(test_polygon_area_two_blobs) { test_polygon_area("testdata/two-blobs.png"); } +MU_TEST(test_polygon_area_two_px) { test_polygon_area("testdata/two-px.png"); } +MU_TEST(test_polygon_area_two_px_inv) { test_polygon_area("testdata/two-px-inv.png"); } + +static void chain_approx_test(const char *fn) { + //cout << endl << "Testing \"" << fn << "\"" << endl; + Image32 ref_img; + mu_assert(ref_img.load(fn), "Input image failed to load"); + ref_img.binarize(); + Image32 ref_img_copy(ref_img); + + TempfileHack tmp_svg(".svg"); + TempfileHack tmp_png(".png"); + + SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows()); + gerbolyze::nopencv::find_contours(ref_img, simplify_contours_teh_chin(ctx.callback())); + ctx.close(); + + mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str())); + + Image32 out_img; + mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load"); + mu_assert_int_eq(ref_img.rows(), out_img.rows()); + mu_assert_int_eq(ref_img.cols(), out_img.cols()); + + double max_abs_deviation = 0; + double rms_sum = 0; + double mean_sum = 0; + for (int y=0; y<out_img.rows(); y++) { + for (int x=0; x<out_img.cols(); x++) { + double delta = fabs((double)out_img.at(x, y)/255.0 - (double)ref_img_copy.at(x, y)); + max_abs_deviation = fmax(max_abs_deviation, delta); + rms_sum += delta*delta; + mean_sum += delta; + } + } + + rms_sum = sqrt(rms_sum / out_img.size()); + mean_sum /= out_img.size(); + if (rms_sum > 0.5) { + snprintf(msg, sizeof(msg), "%s: Chain approximation RMS error is above threshold: %.3f > 0.5\n", fn, rms_sum); + mu_fail(msg); + } + if (mean_sum > 0.1) { + snprintf(msg, sizeof(msg), "%s: Chain approximation mean error is above threshold: %.3f > 0.1\n", fn, mean_sum); + mu_fail(msg); + } + //mu_assert(max_abs_deviation < 0.5, "Maximum chain approximation error is above threshold"); +} + + +MU_TEST(chain_approx_test_chromosome) { chain_approx_test("testdata/chain-approx-teh-chin-chromosome.png"); } +MU_TEST(chain_approx_test_blank) { chain_approx_test("testdata/blank.png"); } +MU_TEST(chain_approx_test_white) { chain_approx_test("testdata/white.png"); } +MU_TEST(chain_approx_test_blob_border_w) { chain_approx_test("testdata/blob-border-w.png"); } +MU_TEST(chain_approx_test_blobs_borders) { chain_approx_test("testdata/blobs-borders.png"); } +MU_TEST(chain_approx_test_blobs_corners) { chain_approx_test("testdata/blobs-corners.png"); } +MU_TEST(chain_approx_test_blobs_crossing) { chain_approx_test("testdata/blobs-crossing.png"); } +MU_TEST(chain_approx_test_cross) { chain_approx_test("testdata/cross.png"); } +MU_TEST(chain_approx_test_letter_e) { chain_approx_test("testdata/letter-e.png"); } +MU_TEST(chain_approx_test_paper_example) { chain_approx_test("testdata/paper-example.png"); } +MU_TEST(chain_approx_test_paper_example_inv) { chain_approx_test("testdata/paper-example-inv.png"); } +MU_TEST(chain_approx_test_single_px) { chain_approx_test("testdata/single-px.png"); } +MU_TEST(chain_approx_test_single_px_inv) { chain_approx_test("testdata/single-px-inv.png"); } +MU_TEST(chain_approx_test_two_blobs) { chain_approx_test("testdata/two-blobs.png"); } +MU_TEST(chain_approx_test_two_px) { chain_approx_test("testdata/two-px.png"); } +MU_TEST(chain_approx_test_two_px_inv) { chain_approx_test("testdata/two-px-inv.png"); } + + +MU_TEST_SUITE(nopencv_contours_suite) { + MU_RUN_TEST(test_complex_example_from_paper); + + MU_RUN_TEST(test_round_trip_blank); + MU_RUN_TEST(test_round_trip_white); + MU_RUN_TEST(test_round_trip_blob_border_w); + MU_RUN_TEST(test_round_trip_blobs_borders); + MU_RUN_TEST(test_round_trip_blobs_corners); + MU_RUN_TEST(test_round_trip_blobs_crossing); + MU_RUN_TEST(test_round_trip_cross); + MU_RUN_TEST(test_round_trip_letter_e); + MU_RUN_TEST(test_round_trip_paper_example); + MU_RUN_TEST(test_round_trip_paper_example_inv); + MU_RUN_TEST(test_round_trip_single_px); + MU_RUN_TEST(test_round_trip_single_px_inv); + MU_RUN_TEST(test_round_trip_two_blobs); + MU_RUN_TEST(test_round_trip_two_px); + MU_RUN_TEST(test_round_trip_two_px_inv); + + MU_RUN_TEST(chain_approx_test_chromosome); + MU_RUN_TEST(chain_approx_test_blank); + MU_RUN_TEST(chain_approx_test_white); + MU_RUN_TEST(chain_approx_test_blob_border_w); + MU_RUN_TEST(chain_approx_test_blobs_borders); + MU_RUN_TEST(chain_approx_test_blobs_corners); + MU_RUN_TEST(chain_approx_test_blobs_crossing); + MU_RUN_TEST(chain_approx_test_cross); + MU_RUN_TEST(chain_approx_test_letter_e); + MU_RUN_TEST(chain_approx_test_paper_example); + MU_RUN_TEST(chain_approx_test_paper_example_inv); + MU_RUN_TEST(chain_approx_test_single_px); + MU_RUN_TEST(chain_approx_test_single_px_inv); + MU_RUN_TEST(chain_approx_test_two_blobs); + MU_RUN_TEST(chain_approx_test_two_px); + MU_RUN_TEST(chain_approx_test_two_px_inv); + + MU_RUN_TEST(test_polygon_area_blank); + MU_RUN_TEST(test_polygon_area_white); + MU_RUN_TEST(test_polygon_area_blob_border_w); + MU_RUN_TEST(test_polygon_area_blobs_borders); + MU_RUN_TEST(test_polygon_area_blobs_corners); + MU_RUN_TEST(test_polygon_area_blobs_crossing); + MU_RUN_TEST(test_polygon_area_cross); + MU_RUN_TEST(test_polygon_area_letter_e); + MU_RUN_TEST(test_polygon_area_paper_example); + MU_RUN_TEST(test_polygon_area_paper_example_inv); + MU_RUN_TEST(test_polygon_area_single_px); + MU_RUN_TEST(test_polygon_area_single_px_inv); + MU_RUN_TEST(test_polygon_area_two_blobs); + MU_RUN_TEST(test_polygon_area_two_px); + MU_RUN_TEST(test_polygon_area_two_px_inv); +}; + +int main(int argc, char **argv) { + (void)argc; + (void)argv; + + MU_RUN_SUITE(nopencv_contours_suite); + MU_REPORT(); + return MU_EXIT_CODE; +} diff --git a/svg-flatten/src/test/svg_tests.py b/svg-flatten/src/test/svg_tests.py new file mode 100644 index 0000000..a5139ec --- /dev/null +++ b/svg-flatten/src/test/svg_tests.py @@ -0,0 +1,75 @@ +#!/usr/bin/env python3 + +import tempfile +import unittest +from pathlib import Path +import subprocess +import os + +from PIL import Image +import numpy as np + +def run_svg_flatten(input_file, output_file, **kwargs): + if 'SVG_FLATTEN' in os.environ: + svg_flatten = os.environ.get('SVG_FLATTEN') + elif (Path(__file__) / '../../build/svg-flatten').is_file(): + svg_flatten = '../../build/svg-flatten' + elif Path('./build/svg-flatten').is_file(): + svg_flatten = './build/svg-flatten' + else: + svg_flatten = 'svg-flatten' + + args = [ svg_flatten, + *(arg for (key, value) in kwargs.items() for arg in (f'--{key.replace("_", "-")}', value)), + str(input_file), str(output_file) ] + + try: + proc = subprocess.run(args, capture_output=True, check=True) + except: + print('Subprocess stdout:') + print(proc.stdout) + print('Subprocess stderr:') + print(proc.stderr) + raise + +class SVGRoundTripTests(unittest.TestCase): + + def compare_images(self, reference, output, test_name, mean=0.01): + ref = np.array(Image.open(reference)) + out = np.array(Image.open(output)) + delta = np.abs(out - ref).astype(float) / 255 + + #print(f'{test_name}: mean={delta.mean():.5g}') + + self.assertTrue(delta.mean() < mean, + f'Expected mean pixel difference between images to be <{mean}, was {delta.mean():.5g}') + + def run_svg_round_trip_test(self, test_in_svg): + with tempfile.NamedTemporaryFile(suffix='.svg') as tmp_out_svg,\ + tempfile.NamedTemporaryFile(suffix='.png') as tmp_out_png,\ + tempfile.NamedTemporaryFile(suffix='.png') as tmp_in_png: + + run_svg_flatten(test_in_svg, tmp_out_svg.name, format='svg') + + subprocess.run(['resvg', tmp_out_svg.name, tmp_out_png.name], check=True, stdout=subprocess.DEVNULL) + subprocess.run(['resvg', test_in_svg, tmp_in_png.name], check=True, stdout=subprocess.DEVNULL) + + try: + self.compare_images(tmp_in_png, tmp_out_png, test_in_svg.stem) + except AssertionError as e: + import shutil + shutil.copyfile(tmp_in_png.name, f'/tmp/gerbolyze-fail-{test_in_svg.stem}-in.png') + shutil.copyfile(tmp_out_png.name, f'/tmp/gerbolyze-fail-{test_in_svg.stem}-out.png') + foo = list(e.args) + foo[0] += '\nFailing test renderings copied to:\n' + foo[0] += f' /tmp/gerbolyze-fail-{test_in_svg.stem}-{{in|out}}.png\n' + e.args = tuple(foo) + raise e + +for test_in_svg in Path('testdata/svg').glob('*.svg'): + # We need to make sure we capture the loop variable's current value here. + gen = lambda testcase: lambda self: self.run_svg_round_trip_test(testcase) + setattr(SVGRoundTripTests, f'test_{test_in_svg.stem}', gen(test_in_svg)) + +if __name__ == '__main__': + unittest.main() diff --git a/svg-flatten/testdata/svg/circles.svg b/svg-flatten/testdata/svg/circles.svg new file mode 100644 index 0000000..9be474e --- /dev/null +++ b/svg-flatten/testdata/svg/circles.svg @@ -0,0 +1,78 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="circles.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.4553" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <circle + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path950" + cx="16.593897" + cy="20.344759" + r="8.5389299" /> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path952" + cx="30.070639" + cy="28.833015" + rx="4.5358982" + ry="8.5728388" /> + <ellipse + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path954" + cx="26.473625" + cy="24.711111" + rx="11.056757" + ry="5.5283785" /> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path952-5" + cx="33.704639" + cy="15.709757" + rx="4.5358982" + ry="8.5728388" /> +</svg> diff --git a/svg-flatten/testdata/svg/compound_xform.svg b/svg-flatten/testdata/svg/compound_xform.svg new file mode 100644 index 0000000..229e05d --- /dev/null +++ b/svg-flatten/testdata/svg/compound_xform.svg @@ -0,0 +1,102 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="compound_xform.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="83.428533" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <g + id="g1200"> + <g + id="g1190" + transform="rotate(-6.212576,25,25)"> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1166" + width="36.952793" + height="36.952793" + x="6.5236034" + y="6.5236034" /> + <rect + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1166-9" + width="32.644421" + height="32.644421" + x="8.6777897" + y="8.6777897" /> + </g> + <g + id="g1164"> + <g + id="g1154" + transform="matrix(-1,0,0,1,50,0)"> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0307622;stroke-linejoin:round;stop-color:#000000" + id="path1130" + cx="47.739796" + cy="33.794945" + rx="14.123164" + ry="19.091663" + transform="matrix(1,0,-0.67287566,0.7397556,0,0)" /> + <rect + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1150" + width="22.3258" + height="20.708179" + x="13.8371" + y="14.64591" /> + </g> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1158" + width="7.5826759" + height="7.5826759" + x="31.564001" + y="-3.791338" + transform="rotate(45)" /> + </g> + </g> +</svg> diff --git a/svg-flatten/testdata/svg/empty.svg b/svg-flatten/testdata/svg/empty.svg new file mode 100644 index 0000000..d2cdc52 --- /dev/null +++ b/svg-flatten/testdata/svg/empty.svg @@ -0,0 +1,11 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8"> +</svg> diff --git a/svg-flatten/testdata/svg/empty_inkscape.svg b/svg-flatten/testdata/svg/empty_inkscape.svg new file mode 100644 index 0000000..716379f --- /dev/null +++ b/svg-flatten/testdata/svg/empty_inkscape.svg @@ -0,0 +1,54 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)" + sodipodi:docname="empty_inkscape.svg"> + <defs + id="defs2" /> + <sodipodi:namedview + id="base" + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1.0" + inkscape:pageopacity="0.0" + inkscape:pageshadow="2" + inkscape:zoom="1.7472574" + inkscape:cx="76.820344" + inkscape:cy="114.80779" + inkscape:document-units="mm" + inkscape:current-layer="layer1" + inkscape:document-rotation="0" + showgrid="false" + inkscape:window-width="1530" + inkscape:window-height="669" + inkscape:window-x="390" + inkscape:window-y="0" + inkscape:window-maximized="0" /> + <metadata + id="metadata5"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <g + inkscape:label="Layer 1" + inkscape:groupmode="layer" + id="layer1" /> +</svg> diff --git a/svg-flatten/testdata/svg/groups.svg b/svg-flatten/testdata/svg/groups.svg new file mode 100644 index 0000000..d94dad0 --- /dev/null +++ b/svg-flatten/testdata/svg/groups.svg @@ -0,0 +1,96 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="groups.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1002" + width="5.4755607" + height="36.937885" + x="22.069475" + y="5.679245" /> + <g + id="g998" + transform="translate(0.10591836,3.2209219)"> + <circle + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path950" + cx="16.593897" + cy="20.344759" + r="8.5389299" /> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path952" + cx="30.070639" + cy="28.833015" + rx="4.5358982" + ry="8.5728388" /> + <ellipse + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path954" + cx="26.473625" + cy="24.711111" + rx="11.056757" + ry="5.5283785" /> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="path952-5" + cx="33.704639" + cy="15.709757" + rx="4.5358982" + ry="8.5728388" /> + </g> + <rect + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1000" + width="31.061703" + height="4.7948809" + x="16.278164" + y="13.511433" /> +</svg> diff --git a/svg-flatten/testdata/svg/pattern_fill.svg b/svg-flatten/testdata/svg/pattern_fill.svg new file mode 100644 index 0000000..21789e6 --- /dev/null +++ b/svg-flatten/testdata/svg/pattern_fill.svg @@ -0,0 +1,135 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:xlink="http://www.w3.org/1999/xlink" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="pattern_fill.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5"> + <pattern + inkscape:collect="always" + xlink:href="#Strips2_1" + id="pattern3491" + patternTransform="matrix(1.774288,0.91540651,-2.3098808,4.4771298,0,1.6295182e-8)" /> + <pattern + inkscape:collect="always" + patternUnits="userSpaceOnUse" + width="1.5" + height="1" + patternTransform="translate(0,0) scale(10,10)" + id="Strips2_1" + inkscape:stockid="Stripes 2:1"> + <rect + style="fill:black;stroke:none" + x="0" + y="-0.5" + width="1" + height="2" + id="rect2399" /> + </pattern> + <pattern + inkscape:collect="always" + xlink:href="#Packedcircles" + id="pattern3461" + patternTransform="matrix(5.9164411,-2.7358919,2.7358917,5.9164404,9.4471894,8.2115447)" /> + <pattern + inkscape:collect="always" + patternUnits="userSpaceOnUse" + width="1" + height="1.73205080756" + patternTransform="translate(0,0) scale(10,10)" + id="Packedcircles" + inkscape:stockid="Packed circles"> + <circle + style="fill:black;stroke:none" + cx="0" + cy="0.5" + r="0.5" + id="circle2421" /> + <circle + style="fill:black;stroke:none" + cx="1" + cy="0.5" + r="0.5" + id="circle2423" /> + <circle + style="fill:black;stroke:none" + cx="0.5" + cy="1.36602540378" + r="0.5" + id="circle2425" /> + <circle + style="fill:black;stroke:none" + cx="0.5" + cy="-0.366025403784" + r="0.5" + id="circle2427" /> + </pattern> + </defs> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.440884" + inkscape:cx="168.05451" + inkscape:cy="88.906258" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <circle + style="opacity:0.99435;fill:url(#pattern3461);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4.00001, 4.00001;stroke-dashoffset:2.19001;stop-color:#000000" + id="path1374" + cx="31.476124" + cy="30.316647" + r="13.079776" /> + <rect + style="opacity:0.99435;fill:url(#pattern3491);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4, 4;stroke-dashoffset:2.19;stop-color:#000000" + id="rect3463" + width="23.181004" + height="23.181004" + x="6.0242319" + y="6.351799" /> + <rect + style="opacity:0.99435;fill:#ffffff;stroke:none;stroke-width:1.16072;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4.64288, 4.64288;stroke-dashoffset:2.54198;stop-color:#000000" + id="rect3493" + width="13.836466" + height="59.941151" + x="40.70414" + y="5.6588583" + transform="matrix(1,0,-0.67013266,0.74224134,0,0)" /> +</svg> diff --git a/svg-flatten/testdata/svg/pattern_stroke.svg b/svg-flatten/testdata/svg/pattern_stroke.svg new file mode 100644 index 0000000..5564df0 --- /dev/null +++ b/svg-flatten/testdata/svg/pattern_stroke.svg @@ -0,0 +1,89 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:xlink="http://www.w3.org/1999/xlink" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="pattern_stroke.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5"> + <pattern + inkscape:collect="always" + xlink:href="#Checkerboard" + id="pattern3520" + patternTransform="scale(1.9742409,1.8180095)" /> + <pattern + inkscape:collect="always" + patternUnits="userSpaceOnUse" + width="2" + height="2" + patternTransform="translate(0,0) scale(10,10)" + id="Checkerboard" + inkscape:stockid="Checkerboard"> + <rect + style="fill:black;stroke:none" + x="0" + y="0" + width="1" + height="1" + id="rect2411" /> + <rect + style="fill:black;stroke:none" + x="1" + y="1" + width="1" + height="1" + id="rect2413" /> + </pattern> + </defs> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.440884" + inkscape:cx="168.05451" + inkscape:cy="88.906258" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <circle + style="opacity:0.99435;fill:none;stroke:url(#pattern3520);stroke-width:8;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:10.95;stroke-opacity:1;stop-color:#000000" + id="path3513" + cx="25" + cy="25" + r="13.90474" /> +</svg> diff --git a/svg-flatten/testdata/svg/pattern_stroke_dashed.svg b/svg-flatten/testdata/svg/pattern_stroke_dashed.svg new file mode 100644 index 0000000..3d8145e --- /dev/null +++ b/svg-flatten/testdata/svg/pattern_stroke_dashed.svg @@ -0,0 +1,89 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:xlink="http://www.w3.org/1999/xlink" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="pattern_stroke_dashed.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5"> + <pattern + inkscape:collect="always" + xlink:href="#Checkerboard" + id="pattern3520" + patternTransform="scale(1.9742409,1.8180095)" /> + <pattern + inkscape:collect="always" + patternUnits="userSpaceOnUse" + width="2" + height="2" + patternTransform="translate(0,0) scale(10,10)" + id="Checkerboard" + inkscape:stockid="Checkerboard"> + <rect + style="fill:black;stroke:none" + x="0" + y="0" + width="1" + height="1" + id="rect2411" /> + <rect + style="fill:black;stroke:none" + x="1" + y="1" + width="1" + height="1" + id="rect2413" /> + </pattern> + </defs> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.440884" + inkscape:cx="168.05451" + inkscape:cy="88.906258" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <circle + style="opacity:0.99435;fill:#000000;stroke:url(#pattern3520);stroke-width:12;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:6,6;stroke-dashoffset:0;stroke-opacity:1;stop-color:#000000" + id="path3513" + cx="25" + cy="25" + r="13.90474" /> +</svg> diff --git a/svg-flatten/testdata/svg/rect.svg b/svg-flatten/testdata/svg/rect.svg new file mode 100644 index 0000000..297ff7e --- /dev/null +++ b/svg-flatten/testdata/svg/rect.svg @@ -0,0 +1,65 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="rect.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="2.6961137" + inkscape:cx="84.54603" + inkscape:cy="109.58858" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832" + width="20" + height="15" + x="10" + y="15" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832-3" + width="20" + height="15" + x="19.318335" + y="22.307203" /> +</svg> diff --git a/svg-flatten/testdata/svg/rect_occlusion.svg b/svg-flatten/testdata/svg/rect_occlusion.svg new file mode 100644 index 0000000..ce0e531 --- /dev/null +++ b/svg-flatten/testdata/svg/rect_occlusion.svg @@ -0,0 +1,79 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="rect_occlusion.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="6.608567" + inkscape:cx="109.89929" + inkscape:cy="79.489935" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832" + width="20" + height="15" + x="10" + y="15" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832-3" + width="20" + height="15" + x="19.318335" + y="22.307203" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832-3-6" + width="20" + height="15" + x="19.318335" + y="22.307203" /> + <rect + style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect832-3-6-7" + width="20" + height="15" + x="23.297699" + y="13.616298" /> +</svg> diff --git a/svg-flatten/testdata/svg/rotation.svg b/svg-flatten/testdata/svg/rotation.svg new file mode 100644 index 0000000..961ef0f --- /dev/null +++ b/svg-flatten/testdata/svg/rotation.svg @@ -0,0 +1,66 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="rotation.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1022" + width="30" + height="10" + x="10" + y="10" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1022-3" + width="30" + height="10" + x="-3.3479581" + y="-44.595863" + transform="rotate(120.85838)" /> +</svg> diff --git a/svg-flatten/testdata/svg/rotation_90.svg b/svg-flatten/testdata/svg/rotation_90.svg new file mode 100644 index 0000000..dcca867 --- /dev/null +++ b/svg-flatten/testdata/svg/rotation_90.svg @@ -0,0 +1,66 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="rotation_90.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1022" + width="30" + height="10" + x="10" + y="10" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000" + id="rect1022-3" + width="30" + height="10" + x="15" + y="-30" + transform="rotate(90)" /> +</svg> diff --git a/svg-flatten/testdata/svg/scale.svg b/svg-flatten/testdata/svg/scale.svg new file mode 100644 index 0000000..91b28d6 --- /dev/null +++ b/svg-flatten/testdata/svg/scale.svg @@ -0,0 +1,91 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="scale.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <g + id="g1082" + transform="matrix(0.81799278,0,0,0.81799278,12.474166,-0.98587389)" + style="stroke-width:1.2225"> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0323454;stroke-linejoin:round;stop-color:#000000" + id="rect1022" + width="30" + height="10" + x="10" + y="10" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0323454;stroke-linejoin:round;stop-color:#000000" + id="rect1022-3" + width="30" + height="10" + x="-3.3479581" + y="-44.595863" + transform="rotate(120.85838)" /> + </g> + <g + id="g1082-5" + transform="matrix(0.61146823,0,0,0.61146823,-0.17038858,14.878729)" + style="stroke-width:1.63541"> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0432701;stroke-linejoin:round;stop-color:#000000" + id="rect1022-6" + width="30" + height="10" + x="10" + y="10" /> + <rect + style="fill:#000000;stroke:none;stroke-width:0.0432701;stroke-linejoin:round;stop-color:#000000" + id="rect1022-3-2" + width="30" + height="10" + x="-3.3479581" + y="-44.595863" + transform="rotate(120.85838)" /> + </g> +</svg> diff --git a/svg-flatten/testdata/svg/shear.svg b/svg-flatten/testdata/svg/shear.svg new file mode 100644 index 0000000..ec7875f --- /dev/null +++ b/svg-flatten/testdata/svg/shear.svg @@ -0,0 +1,59 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="shear.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="97.954218" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" /> + <ellipse + style="fill:#000000;stroke:none;stroke-width:0.0307622;stroke-linejoin:round;stop-color:#000000" + id="path1130" + cx="47.739796" + cy="33.794945" + rx="14.123164" + ry="19.091663" + transform="matrix(1,0,-0.67287566,0.7397556,0,0)" /> +</svg> diff --git a/svg-flatten/testdata/svg/stroke.svg b/svg-flatten/testdata/svg/stroke.svg new file mode 100644 index 0000000..6a4b334 --- /dev/null +++ b/svg-flatten/testdata/svg/stroke.svg @@ -0,0 +1,67 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="stroke.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.2278448" + inkscape:cx="83.06317" + inkscape:cy="83.428533" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <rect + style="fill:#ffffff;stroke:#000000;stroke-width:9.99998;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stop-color:#000000" + id="rect1220" + width="30" + height="30.000029" + x="10" + y="10" /> + <rect + style="fill:#ffffff;stroke:#000000;stroke-width:2;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stop-color:#000000" + id="rect1220-1" + width="11" + height="11" + x="19.5" + y="19.5" /> +</svg> diff --git a/svg-flatten/testdata/svg/stroke_caps.svg b/svg-flatten/testdata/svg/stroke_caps.svg new file mode 100644 index 0000000..6b89a34 --- /dev/null +++ b/svg-flatten/testdata/svg/stroke_caps.svg @@ -0,0 +1,68 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="stroke_caps.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.4020648" + inkscape:cx="99.189726" + inkscape:cy="80.388351" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:none" + d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404" + id="path1278" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404" + id="path1278-2" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:square;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404" + id="path1278-7" + sodipodi:nodetypes="cccc" /> +</svg> diff --git a/svg-flatten/testdata/svg/stroke_dashes.svg b/svg-flatten/testdata/svg/stroke_dashes.svg new file mode 100644 index 0000000..c1d6f5c --- /dev/null +++ b/svg-flatten/testdata/svg/stroke_dashes.svg @@ -0,0 +1,75 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="stroke_dashes.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.4020648" + inkscape:cx="99.189726" + inkscape:cy="80.388351" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <path + style="fill:none;stroke:#000000;stroke-width:3.00000008;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:3.00000008,3.00000008;stroke-dashoffset:0" + d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404" + id="path1278" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:2, 1, 0.50000000000000000, 1;stroke-opacity:1;stroke-dashoffset:0;opacity:0.99435036" + d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404" + id="path1278-2" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:0.1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:0.4,0.4;stroke-opacity:1;stroke-dashoffset:0" + d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404" + id="path1278-7" + sodipodi:nodetypes="cccc" /> + <rect + style="opacity:0.99435;fill:none;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4, 4;stroke-dashoffset:2.19;stop-color:#000000" + id="rect1354" + width="43.84502" + height="43.932682" + x="3.0774899" + y="3.033659" /> +</svg> diff --git a/svg-flatten/testdata/svg/stroke_joins.svg b/svg-flatten/testdata/svg/stroke_joins.svg new file mode 100644 index 0000000..ee3e1ac --- /dev/null +++ b/svg-flatten/testdata/svg/stroke_joins.svg @@ -0,0 +1,68 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="stroke_joins.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5" /> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.4020648" + inkscape:cx="99.189726" + inkscape:cy="80.388351" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:none" + d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404" + id="path1278" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:bevel;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404" + id="path1278-2" + sodipodi:nodetypes="cccc" /> + <path + style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" + d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404" + id="path1278-7" + sodipodi:nodetypes="cccc" /> +</svg> diff --git a/svg-flatten/testdata/svg/text.svg b/svg-flatten/testdata/svg/text.svg new file mode 100644 index 0000000..bd8c390 --- /dev/null +++ b/svg-flatten/testdata/svg/text.svg @@ -0,0 +1,108 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" + xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + width="50mm" + height="50mm" + viewBox="0 0 50 50" + version="1.1" + id="svg8" + sodipodi:docname="text.svg" + inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"> + <metadata + id="metadata7"> + <rdf:RDF> + <cc:Work + rdf:about=""> + <dc:format>image/svg+xml</dc:format> + <dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> + <dc:title></dc:title> + </cc:Work> + </rdf:RDF> + </metadata> + <defs + id="defs5"> + <rect + x="3.0026957" + y="3.2239678" + width="65.46684" + height="67.27124" + id="rect3560" /> + <rect + x="3.0026958" + y="3.2239678" + width="65.466843" + height="67.27124" + id="rect3560-2" /> + <rect + x="3.0026958" + y="3.2239678" + width="65.466843" + height="67.27124" + id="rect3573" /> + <rect + x="3.0026958" + y="3.2239678" + width="65.466843" + height="67.27124" + id="rect3560-2-4" /> + <rect + x="3.0026958" + y="3.2239678" + width="65.466843" + height="67.27124" + id="rect3624" /> + </defs> + <sodipodi:namedview + pagecolor="#ffffff" + bordercolor="#666666" + borderopacity="1" + objecttolerance="10" + gridtolerance="10" + guidetolerance="10" + inkscape:pageopacity="0" + inkscape:pageshadow="2" + inkscape:window-width="1920" + inkscape:window-height="1019" + id="namedview3" + showgrid="false" + inkscape:zoom="3.440884" + inkscape:cx="92.105039" + inkscape:cy="86.46986" + inkscape:window-x="0" + inkscape:window-y="0" + inkscape:window-maximized="1" + inkscape:current-layer="svg8" + inkscape:document-units="mm" + showguides="false" /> + <text + xml:space="preserve" + id="text3558" + style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560);" + transform="translate(4.917252,17.244243)"><tspan + x="3.0019531" + y="9.5799363"><tspan + style="font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text> + <text + xml:space="preserve" + id="text3558-0" + style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560-2);" + transform="matrix(1,-0.42912384,0,1,4.8672981,38.568218)"><tspan + x="3.0019531" + y="9.5799363"><tspan + style="font-style:italic;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text> + <text + xml:space="preserve" + id="text3558-0-1" + style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560-2-4);" + transform="matrix(1,-0.42912384,0,1,4.8672981,13.281795)"><tspan + x="3.0019531" + y="9.5799363"><tspan + style="font-style:italic;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text> +</svg> -- cgit