#include #include "libslic3r/Point.hpp" #include "libslic3r/Polygon.hpp" using namespace Slic3r; SCENARIO("Converted Perl tests", "[Polygon]") { GIVEN("ccw_square") { Polygon ccw_square{ { 100, 100 }, { 200, 100 }, { 200, 200 }, { 100, 200 } }; Polygon cw_square(ccw_square); cw_square.reverse(); THEN("ccw_square is valid") { REQUIRE(ccw_square.is_valid()); } THEN("cw_square is valid") { REQUIRE(cw_square.is_valid()); } THEN("ccw_square.area") { REQUIRE(ccw_square.area() == 100 * 100); } THEN("cw_square.area") { REQUIRE(cw_square.area() == - 100 * 100); } THEN("ccw_square.centroid") { REQUIRE(ccw_square.centroid() == Point { 150, 150 }); } THEN("cw_square.centroid") { REQUIRE(cw_square.centroid() == Point { 150, 150 }); } THEN("ccw_square.contains_point(150, 150)") { REQUIRE(ccw_square.contains({ 150, 150 })); } THEN("cw_square.contains_point(150, 150)") { REQUIRE(cw_square.contains({ 150, 150 })); } THEN("conversion to lines") { REQUIRE(ccw_square.lines() == Lines{ { { 100, 100 }, { 200, 100 } }, { { 200, 100 }, { 200, 200 } }, { { 200, 200 }, { 100, 200 } }, { { 100, 200 }, { 100, 100 } } }); } THEN("split_at_first_point") { REQUIRE(ccw_square.split_at_first_point() == Polyline { ccw_square[0], ccw_square[1], ccw_square[2], ccw_square[3], ccw_square[0] }); } THEN("split_at_index(2)") { REQUIRE(ccw_square.split_at_index(2) == Polyline { ccw_square[2], ccw_square[3], ccw_square[0], ccw_square[1], ccw_square[2] }); } THEN("split_at_vertex(ccw_square[2])") { REQUIRE(ccw_square.split_at_vertex(ccw_square[2]) == Polyline { ccw_square[2], ccw_square[3], ccw_square[0], ccw_square[1], ccw_square[2] }); } THEN("is_counter_clockwise") { REQUIRE(ccw_square.is_counter_clockwise()); } THEN("! is_counter_clockwise") { REQUIRE(! cw_square.is_counter_clockwise()); } THEN("make_counter_clockwise") { cw_square.make_counter_clockwise(); REQUIRE(cw_square.is_counter_clockwise()); } THEN("make_counter_clockwise^2") { cw_square.make_counter_clockwise(); cw_square.make_counter_clockwise(); REQUIRE(cw_square.is_counter_clockwise()); } THEN("first_point") { REQUIRE(&ccw_square.first_point() == &ccw_square.points.front()); } } GIVEN("Triangulating hexagon") { Polygon hexagon{ { 100, 0 } }; for (size_t i = 1; i < 6; ++ i) { Point p = hexagon.points.front(); p.rotate(PI / 3 * i); hexagon.points.emplace_back(p); } Polygons triangles; hexagon.triangulate_convex(&triangles); THEN("right number of triangles") { REQUIRE(triangles.size() == 4); } THEN("all triangles are ccw") { auto it = std::find_if(triangles.begin(), triangles.end(), [](const Polygon &tri) { return tri.is_clockwise(); }); REQUIRE(it == triangles.end()); } } GIVEN("General triangle") { Polygon polygon { { 50000000, 100000000 }, { 300000000, 102000000 }, { 50000000, 104000000 } }; Line line { { 175992032, 102000000 }, { 47983964, 102000000 } }; Point intersection; bool has_intersection = polygon.intersection(line, &intersection); THEN("Intersection with line") { REQUIRE(has_intersection); REQUIRE(intersection == Point { 50000000, 102000000 }); } } } TEST_CASE("Centroid of Trapezoid must be inside", "[Polygon][Utils]") { Slic3r::Polygon trapezoid { { 4702134, 1124765853 }, { -4702134, 1124765853 }, { -9404268, 1049531706 }, { 9404268, 1049531706 }, }; Point centroid = trapezoid.centroid(); CHECK(trapezoid.contains(centroid)); } // This test currently only covers remove_collinear_points. // All remaining tests are to be ported from xs/t/06_polygon.t Slic3r::Points collinear_circle({ Slic3r::Point::new_scale(0, 0), // 3 collinear points at beginning Slic3r::Point::new_scale(10, 0), Slic3r::Point::new_scale(20, 0), Slic3r::Point::new_scale(30, 10), Slic3r::Point::new_scale(40, 20), // 2 collinear points Slic3r::Point::new_scale(40, 30), Slic3r::Point::new_scale(30, 40), // 3 collinear points Slic3r::Point::new_scale(20, 40), Slic3r::Point::new_scale(10, 40), Slic3r::Point::new_scale(-10, 20), Slic3r::Point::new_scale(-20, 10), Slic3r::Point::new_scale(-20, 0), // 3 collinear points at end Slic3r::Point::new_scale(-10, 0), Slic3r::Point::new_scale(-5, 0) }); SCENARIO("Remove collinear points from Polygon", "[Polygon]") { GIVEN("Polygon with collinear points"){ Slic3r::Polygon p(collinear_circle); WHEN("collinear points are removed") { remove_collinear(p); THEN("Leading collinear points are removed") { REQUIRE(p.points.front() == Slic3r::Point::new_scale(20, 0)); } THEN("Trailing collinear points are removed") { REQUIRE(p.points.back() == Slic3r::Point::new_scale(-20, 0)); } THEN("Number of remaining points is correct") { REQUIRE(p.points.size() == 7); } } } } SCENARIO("Simplify polygon", "[Polygon]") { GIVEN("gear") { auto gear = Polygon::new_scale({ {144.9694,317.1543}, {145.4181,301.5633}, {146.3466,296.921}, {131.8436,294.1643}, {131.7467,294.1464}, {121.7238,291.5082}, {117.1631,290.2776}, {107.9198,308.2068}, {100.1735,304.5101}, {104.9896,290.3672}, {106.6511,286.2133}, {93.453,279.2327}, {81.0065,271.4171}, {67.7886,286.5055}, {60.7927,280.1127}, {69.3928,268.2566}, {72.7271,264.9224}, {61.8152,253.9959}, {52.2273,242.8494}, {47.5799,245.7224}, {34.6577,252.6559}, {30.3369,245.2236}, {42.1712,236.3251}, {46.1122,233.9605}, {43.2099,228.4876}, {35.0862,211.5672}, {33.1441,207.0856}, {13.3923,212.1895}, {10.6572,203.3273}, {6.0707,204.8561}, {7.2775,204.4259}, {29.6713,196.3631}, {25.9815,172.1277}, {25.4589,167.2745}, {19.8337,167.0129}, {5.0625,166.3346}, {5.0625,156.9425}, {5.3701,156.9282}, {21.8636,156.1628}, {25.3713,156.4613}, {25.4243,155.9976}, {29.3432,155.8157}, {30.3838,149.3549}, {26.3596,147.8137}, {27.1085,141.2604}, {29.8466,126.8337}, {24.5841,124.9201}, {10.6664,119.8989}, {13.4454,110.9264}, {33.1886,116.0691}, {38.817,103.1819}, {45.8311,89.8133}, {30.4286,76.81}, {35.7686,70.0812}, {48.0879,77.6873}, {51.564,81.1635}, {61.9006,69.1791}, {72.3019,58.7916}, {60.5509,42.5416}, {68.3369,37.1532}, {77.9524,48.1338}, {80.405,52.2215}, {92.5632,44.5992}, {93.0123,44.3223}, {106.3561,37.2056}, {100.8631,17.4679}, {108.759,14.3778}, {107.3148,11.1283}, {117.0002,32.8627}, {140.9109,27.3974}, {145.7004,26.4994}, {145.1346,6.1011}, {154.502,5.4063}, {156.9398,25.6501}, {171.0557,26.2017}, {181.3139,27.323}, {186.2377,27.8532}, {191.6031,8.5474}, {200.6724,11.2756}, {197.2362,30.2334}, {220.0789,39.1906}, {224.3261,41.031}, {236.3506,24.4291}, {243.6897,28.6723}, {234.2956,46.7747}, {245.6562,55.1643}, {257.2523,65.0901}, {261.4374,61.5679}, {273.1709,52.8031}, {278.555,59.5164}, {268.4334,69.8001}, {264.1615,72.3633}, {268.2763,77.9442}, {278.8488,93.5305}, {281.4596,97.6332}, {286.4487,95.5191}, {300.2821,90.5903}, {303.4456,98.5849}, {286.4523,107.7253}, {293.7063,131.1779}, {294.9748,135.8787}, {314.918,133.8172}, {315.6941,143.2589}, {300.9234,146.1746}, {296.6419,147.0309}, {297.1839,161.7052}, {296.6136,176.3942}, {302.1147,177.4857}, {316.603,180.3608}, {317.1658,176.7341}, {315.215,189.6589}, {315.1749,189.6548}, {294.9411,187.5222}, {291.13,201.7233}, {286.2615,215.5916}, {291.1944,218.2545}, {303.9158,225.1271}, {299.2384,233.3694}, {285.7165,227.6001}, {281.7091,225.1956}, {273.8981,237.6457}, {268.3486,245.2248}, {267.4538,246.4414}, {264.8496,250.0221}, {268.6392,253.896}, {278.5017,265.2131}, {272.721,271.4403}, {257.2776,258.3579}, {234.4345,276.5687}, {242.6222,294.8315}, {234.9061,298.5798}, {227.0321,286.2841}, {225.2505,281.8301}, {211.5387,287.8187}, {202.3025,291.0935}, {197.307,292.831}, {199.808,313.1906}, {191.5298,315.0787}, {187.3082,299.8172}, {186.4201,295.3766}, {180.595,296.0487}, {161.7854,297.4248}, {156.8058,297.6214}, {154.3395,317.8592} }); WHEN("simplified") { size_t num_points = gear.size(); Polygons simplified = gear.simplify(1000.); THEN("gear simplified to a single polygon") { REQUIRE(simplified.size() == 1); } THEN("gear was reduced using Douglas-Peucker") { //note printf "original points: %d\nnew points: %d", $num_points, scalar(@{$simplified->[0]}); REQUIRE(simplified.front().size() < num_points); } } } GIVEN("hole in square") { // CW oriented auto hole_in_square = Polygon{ {140, 140}, {140, 160}, {160, 160}, {160, 140} }; WHEN("simplified") { Polygons simplified = hole_in_square.simplify(2.); THEN("hole simplification returns one polygon") { REQUIRE(simplified.size() == 1); } THEN("hole simplification turns cw polygon into ccw polygon") { REQUIRE(simplified.front().is_counter_clockwise()); } } } }