diff --git a/geometry/geometry.pro b/geometry/geometry.pro index eaeb7e42b8..0a97c2fefd 100644 --- a/geometry/geometry.pro +++ b/geometry/geometry.pro @@ -28,3 +28,4 @@ HEADERS += \ simplification.hpp \ transformations.hpp \ tree4d.hpp \ + polygon.hpp \ diff --git a/geometry/geometry_tests/geometry_tests.pro b/geometry/geometry_tests/geometry_tests.pro index 322c5d1cb8..ba26c8c323 100644 --- a/geometry/geometry_tests/geometry_tests.pro +++ b/geometry/geometry_tests/geometry_tests.pro @@ -29,3 +29,4 @@ SOURCES += \ simplification_test.cpp \ transformations_test.cpp \ tree_test.cpp \ + polygon_test.cpp \ diff --git a/geometry/geometry_tests/polygon_test.cpp b/geometry/geometry_tests/polygon_test.cpp new file mode 100644 index 0000000000..c877e93057 --- /dev/null +++ b/geometry/geometry_tests/polygon_test.cpp @@ -0,0 +1,61 @@ +#include "../../testing/testing.hpp" +#include "../polygon.hpp" +#include "../point2d.hpp" +#include "../../base/macros.hpp" + +namespace +{ +typedef m2::PointD P; +} + +UNIT_TEST(IsSegmentInCone) +{ + TEST(IsSegmentInCone(P(0,0), P( 0, 3), P(-1,-1), P(1,-1)), ()); + TEST(IsSegmentInCone(P(0,0), P( 2, 3), P(-1,-1), P(1,-1)), ()); + TEST(IsSegmentInCone(P(0,0), P(-3, 3), P(-1,-1), P(1,-1)), ()); + TEST(IsSegmentInCone(P(0,0), P(-3, 0), P(-1,-1), P(1,-1)), ()); + TEST(IsSegmentInCone(P(0,0), P( 3, 0), P(-1,-1), P(1,-1)), ()); + TEST(!IsSegmentInCone(P(0,0), P( 0,-1), P(-1,-1), P(1,-1)), ()); + TEST(!IsSegmentInCone(P(0,0), P( 1,-3), P(-1,-1), P(1,-1)), ()); + TEST(!IsSegmentInCone(P(0,0), P(-1,-3), P(-1,-1), P(1,-1)), ()); + + TEST(IsSegmentInCone(P(0,0), P( 0, 3), P(-1,1), P(1,1)), ()); + TEST(IsSegmentInCone(P(0,0), P( 2, 3), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P(-3, 3), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P(-3, 0), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P( 3, 0), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P( 0,-1), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P( 1,-3), P(-1,1), P(1,1)), ()); + TEST(!IsSegmentInCone(P(0,0), P(-1,-3), P(-1,1), P(1,1)), ()); +} + +UNIT_TEST(IsDiagonalVisible) +{ + P polyA [] = { P(0,0), P(3,0), P(3,2), P(2,2), P(2,1), P(0,1) }; + vector
poly(&polyA[0], &polyA[0] + ARRAY_SIZE(polyA)); + // TODO: Reverse directions. + + TEST(!IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 0, poly.begin() + 2), ()); + TEST(!IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 0, poly.begin() + 3), ()); + TEST( IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 0, poly.begin() + 4), ()); + TEST(!IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 5, poly.begin() + 3), ()); + TEST(!IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 5, poly.begin() + 2), ()); + TEST( IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 5, poly.begin() + 1), ()); + TEST( IsDiagonalVisible(poly.begin(), poly.end(), poly.begin() + 1, poly.begin() + 5), ()); +} + +UNIT_TEST(FindSingleStrip) +{ + { + P poly [] = { P(0,0), P(3,0), P(3,2), P(2,2), P(2,1), P(0,1) }; + size_t const n = ARRAY_SIZE(poly); + TEST_NOT_EQUAL( + FindSingleStrip(n, IsDiagonalVisibleFunctor
(&poly[0], &poly[0] + n)), n, ()); + } + { + P poly [] = { P(0,0), P(2, -1), P(3,-1), P(3,2), P(2,2), P(2,1), P(0,1) }; + size_t const n = ARRAY_SIZE(poly); + TEST_EQUAL( + FindSingleStrip(n, IsDiagonalVisibleFunctor
(&poly[0], &poly[0] + n)), n, ());
+ }
+}
diff --git a/geometry/polygon.hpp b/geometry/polygon.hpp
new file mode 100644
index 0000000000..bb1e63245c
--- /dev/null
+++ b/geometry/polygon.hpp
@@ -0,0 +1,82 @@
+#pragma once
+
+#include "../base/assert.hpp"
+#include "../base/base.hpp"
+#include "../base/math.hpp"
+#include "../base/stl_add.hpp"
+
+// If polygon with n vertices is a single strip, return the start index of the strip or n otherwise.
+template