From 8a1166c7f8e31240ec61f342ee84f4632bf4378c Mon Sep 17 00:00:00 2001 From: Arsentiy Milchakov Date: Wed, 6 May 2020 11:57:56 +0300 Subject: [PATCH] [geometry] intersection score is added --- editor/feature_matcher.cpp | 102 +++--------- geometry/CMakeLists.txt | 1 + geometry/geometry_tests/CMakeLists.txt | 1 + .../intersection_score_tests.cpp | 95 +++++++++++ geometry/intersection_score.hpp | 150 ++++++++++++++++++ .../geometry.xcodeproj/project.pbxproj | 28 +++- 6 files changed, 290 insertions(+), 87 deletions(-) create mode 100644 geometry/geometry_tests/intersection_score_tests.cpp create mode 100644 geometry/intersection_score.hpp diff --git a/editor/feature_matcher.cpp b/editor/feature_matcher.cpp index 6b30d570b3..549089c05e 100644 --- a/editor/feature_matcher.cpp +++ b/editor/feature_matcher.cpp @@ -1,5 +1,7 @@ #include "editor/feature_matcher.hpp" +#include "geometry/intersection_score.hpp" + #include "base/logging.hpp" #include "base/stl_helpers.hpp" #include "base/stl_iterator.hpp" @@ -34,18 +36,6 @@ using ForEachRefFn = function; using ForEachWayFn = function; double const kPointDiffEps = 1e-5; -double const kPenaltyScore = -1; - -template -AreaType IntersectionArea(LGeometry const & our, RGeometry const & their) -{ - ASSERT(bg::is_valid(our), ()); - ASSERT(bg::is_valid(their), ()); - - MultiPolygon result; - bg::intersection(our, their, result); - return bg::area(result); -} void AddInnerIfNeeded(pugi::xml_document const & osmResponse, pugi::xml_node const & way, Polygon & dest) @@ -85,64 +75,6 @@ void MakeOuterRing(MultiLinestring & outerLines, Polygon & dest) } } -template -double MatchByGeometry(LGeometry const & lhs, RGeometry const & rhs) -{ - if (!bg::is_valid(lhs) || !bg::is_valid(rhs)) - return kPenaltyScore; - - auto const lhsArea = bg::area(lhs); - auto const rhsArea = bg::area(rhs); - auto const intersectionArea = IntersectionArea(lhs, rhs); - auto const unionArea = lhsArea + rhsArea - intersectionArea; - - // Avoid infinity. - if (base::AlmostEqualAbs(unionArea, 0.0, 1e-18)) - return kPenaltyScore; - - auto const score = intersectionArea / unionArea; - - // If area of the intersection is a half of the object area, penalty score will be returned. - if (score <= 0.5) - return kPenaltyScore; - - return score; -} - -MultiPolygon TrianglesToPolygon(vector const & points) -{ - size_t const kTriangleSize = 3; - if (points.size() % kTriangleSize != 0) - MYTHROW(matcher::NotAPolygonException, ("Count of points must be multiple of", kTriangleSize)); - - vector polygons; - for (size_t i = 0; i < points.size(); i += kTriangleSize) - { - MultiPolygon polygon; - polygon.resize(1); - auto & p = polygon[0]; - auto & outer = p.outer(); - for (size_t j = i; j < i + kTriangleSize; ++j) - outer.push_back(PointXY(points[j].x, points[j].y)); - bg::correct(p); - if (!bg::is_valid(polygon)) - MYTHROW(matcher::NotAPolygonException, ("The triangle is not valid")); - polygons.push_back(polygon); - } - - if (polygons.empty()) - return {}; - - auto & result = polygons[0]; - for (size_t i = 1; i < polygons.size(); ++i) - { - MultiPolygon u; - bg::union_(result, polygons[i], u); - u.swap(result); - } - return result; -} - /// Returns value form (-Inf, 1]. Negative values are used as penalty, positive as score. double ScoreLatLon(XMLFeature const & xmlFt, ms::LatLon const & latLon) { @@ -268,14 +200,20 @@ double ScoreGeometry(pugi::xml_document const & osmResponse, pugi::xml_node cons auto const their = GetWaysOrRelationsGeometry(osmResponse, wayOrRelation); if (bg::is_empty(their)) - return kPenaltyScore; + return geometry::kPenaltyScore; - auto const our = TrianglesToPolygon(ourGeometry); + auto const our = geometry::TrianglesToPolygon(ourGeometry); if (bg::is_empty(our)) - return kPenaltyScore; + return geometry::kPenaltyScore; - return MatchByGeometry(our, their); + auto const score = geometry::GetIntersectionScore(our, their); + + // If area of the intersection is a half of the object area, penalty score will be returned. + if (score <= 0.5) + return geometry::kPenaltyScore; + + return score; } } // namespace @@ -283,7 +221,7 @@ namespace matcher { pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon const & latLon) { - double bestScore = kPenaltyScore; + double bestScore = geometry::kPenaltyScore; pugi::xml_node bestMatchNode; for (auto const & xNode : osmResponse.select_nodes("osm/node")) @@ -319,7 +257,7 @@ pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse, vector const & geometry) { - double bestScore = kPenaltyScore; + double bestScore = geometry::kPenaltyScore; pugi::xml_node bestMatchWay; auto const xpath = "osm/way|osm/relation[tag[@k='type' and @v='multipolygon']]"; @@ -342,15 +280,13 @@ pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse, double ScoreTriangulatedGeometries(vector const & lhs, vector const & rhs) { - auto const lhsPolygon = TrianglesToPolygon(lhs); - if (bg::is_empty(lhsPolygon)) - return kPenaltyScore; + auto const score = geometry::GetIntersectionScoreForTriangulated(lhs, rhs); - auto const rhsPolygon = TrianglesToPolygon(rhs); - if (bg::is_empty(rhsPolygon)) - return kPenaltyScore; + // If area of the intersection is a half of the object area, penalty score will be returned. + if (score <= 0.5) + return geometry::kPenaltyScore; - return MatchByGeometry(lhsPolygon, rhsPolygon); + return score; } double ScoreTriangulatedGeometriesByPoints(vector const & lhs, diff --git a/geometry/CMakeLists.txt b/geometry/CMakeLists.txt index 8dcc14c909..bd79b39286 100644 --- a/geometry/CMakeLists.txt +++ b/geometry/CMakeLists.txt @@ -46,6 +46,7 @@ set( point_with_altitude.cpp point_with_altitude.hpp polygon.hpp + intersection_score.hpp polyline2d.hpp rect2d.hpp rect_intersect.hpp diff --git a/geometry/geometry_tests/CMakeLists.txt b/geometry/geometry_tests/CMakeLists.txt index c5a5d57b4f..1e0ae83829 100644 --- a/geometry/geometry_tests/CMakeLists.txt +++ b/geometry/geometry_tests/CMakeLists.txt @@ -20,6 +20,7 @@ set( distance_on_sphere_test.cpp equality.hpp intersect_test.cpp + intersection_score_tests.cpp large_polygon.hpp latlon_test.cpp line2d_tests.cpp diff --git a/geometry/geometry_tests/intersection_score_tests.cpp b/geometry/geometry_tests/intersection_score_tests.cpp new file mode 100644 index 0000000000..0e600966d5 --- /dev/null +++ b/geometry/geometry_tests/intersection_score_tests.cpp @@ -0,0 +1,95 @@ +#include "testing/testing.hpp" + +#include "geometry/any_rect2d.hpp" +#include "geometry/intersection_score.hpp" +#include "geometry/point2d.hpp" +#include "geometry/rect2d.hpp" + +#include "base/math.hpp" + +#include + +UNIT_TEST(IntersectionScore_PointsToPolygon) +{ + { + m2::RectD rectD = {0, 0, 10, 10}; + m2::AnyRectD const anyRect1(rectD); + rectD = {0, 0, 10, 9}; + m2::AnyRectD const anyRect2(rectD); + + m2::AnyRectD::Corners corners1; + anyRect1.GetGlobalPoints(corners1); + + m2::AnyRectD::Corners corners2; + anyRect2.GetGlobalPoints(corners2); + + auto const score = geometry::GetIntersectionScoreForPoints(corners1, corners2); + + TEST(base::AlmostEqualAbs(score, 0.9, 1e-10), ()); + } + { + m2::RectD rectD = {0, 0, 10, 10}; + m2::AnyRectD const anyRect1(rectD); + rectD = {10, 10, 20, 20}; + m2::AnyRectD const anyRect2(rectD); + + m2::AnyRectD::Corners corners1; + anyRect1.GetGlobalPoints(corners1); + + m2::AnyRectD::Corners corners2; + anyRect2.GetGlobalPoints(corners2); + + auto const score = geometry::GetIntersectionScoreForPoints(corners1, corners2); + + TEST(base::AlmostEqualAbs(score, 0.0, 1e-10), ()); + } + { + m2::RectD rectD = {0, 0, 10, 10}; + m2::AnyRectD const anyRect1(rectD); + m2::AnyRectD::Corners corners1; + anyRect1.GetGlobalPoints(corners1); + + // Backward + m2::AnyRectD::Corners corners2 = {m2::PointD{10.0, 10.0}, {10.0, 0.0}, {0.0, 0.0}, {0.0, 10.0}}; + auto const score = geometry::GetIntersectionScoreForPoints(corners1, corners2); + + TEST(base::AlmostEqualAbs(score, 1.0, 1e-10), ()); + } +} + +UNIT_TEST(IntersectionScore_TrianglesToPolygon) +{ + { + std::vector triangiulated1 = {{0.0, 0.0}, {0.0, 10.0}, {10.0, 0.0}, + {10.0, 0.0}, {0.0, 10.0}, {10.0, 10.0}}; + std::vector triangiulated2 = {{0.0, 0.0}, {0.0, 9.0}, {10.0, 0.0}, + {10.0, 0.0}, {0.0, 9.0}, {10.0, 9.0}}; + + auto const score = + geometry::GetIntersectionScoreForTriangulated(triangiulated1, triangiulated2); + + TEST(base::AlmostEqualAbs(score, 0.9, 1e-10), ()); + } + { + m2::RectD rectD = {0, 0, 10, 10}; + m2::AnyRectD const anyRect1(rectD); + rectD = {10, 10, 20, 20}; + m2::AnyRectD const anyRect2(rectD); + + m2::AnyRectD::Corners corners1; + anyRect1.GetGlobalPoints(corners1); + + m2::AnyRectD::Corners corners2; + anyRect2.GetGlobalPoints(corners2); + + std::vector triangiulated1 = {{0.0, 0.0}, {0.0, 10.0}, {10.0, 0.0}, + {10.0, 0.0}, {0.0, 10.0}, {10.0, 10.0}}; + std::vector triangiulated2 = {{10.0, 10.0}, {10.0, 20.0}, {20.0, 10.0}, + {20.0, 10.0}, {10.0, 20.0}, {20.0, 20.0}}; + + auto const score = + geometry::GetIntersectionScoreForTriangulated(triangiulated1, triangiulated2); + + TEST(base::AlmostEqualAbs(score, 0.0, 1e-10), ()); + } +} diff --git a/geometry/intersection_score.hpp b/geometry/intersection_score.hpp new file mode 100644 index 0000000000..d024939df0 --- /dev/null +++ b/geometry/intersection_score.hpp @@ -0,0 +1,150 @@ +#pragma once + +#include "geometry/any_rect2d.hpp" +#include "geometry/point2d.hpp" + +#include "base/exception.hpp" +#include "base/math.hpp" + +#include + +#include +#include +#include +#include + +namespace geometry +{ +double constexpr kPenaltyScore = -1.0; +DECLARE_EXCEPTION(NotAPolygonException, RootException); + +namespace impl +{ +using PointXY = boost::geometry::model::d2::point_xy; +using Polygon = boost::geometry::model::polygon; +using MultiPolygon = boost::geometry::model::multi_polygon; +} // namespace impl + +template +impl::Polygon PointsToPolygon(Container const & points); +template +impl::MultiPolygon TrianglesToPolygon(Container const & points); + +/// Returns value form [-1.0, 1.0]. +/// Returns positive value when projections of two geometries are +/// intersects (returns intersection area divided by union area). +/// Returns zero when projections of two geometries are not intersects. +/// Returns 1 when provided geometries projections are fully equal. +/// Returns -1.0 as penalty. It is possible when any of the geometries is empty or not valid. +/// |lhs| and |rhs| are any areal boost::geometry types. +template +double GetIntersectionScore(LPolygon const & lhs, RPolygon const & rhs) +{ + if (!boost::geometry::is_valid(lhs) || !boost::geometry::is_valid(rhs) || + boost::geometry::is_empty(lhs) || boost::geometry::is_empty(rhs)) + { + return kPenaltyScore; + } + + auto const lhsArea = boost::geometry::area(lhs); + auto const rhsArea = boost::geometry::area(rhs); + impl::MultiPolygon result; + boost::geometry::intersection(lhs, rhs, result); + auto const intersectionArea = boost::geometry::area(result); + auto const unionArea = lhsArea + rhsArea - intersectionArea; + + auto const score = intersectionArea / unionArea; + + return score; +} + +/// Throws NotAPolygonException exception. +/// For detailed info see comment for +/// double GetIntersectionScore(LPolygon const & lhs, RPolygon const & rhs). +/// |lhs| and |rhs| are any standard container of m2::Point with random access iterator. +/// |toPolygonConverter| is a method which converts |lhs| and |rhs| to boost::geometry aerial type. +template +double GetIntersectionScore(Container const & lhs, Container const & rhs, + Converter const & toPolygonConverter) +{ + auto const lhsPolygon = toPolygonConverter(lhs); + if (boost::geometry::is_empty(lhsPolygon)) + return kPenaltyScore; + + auto const rhsPolygon = toPolygonConverter(rhs); + if (boost::geometry::is_empty(rhsPolygon)) + return kPenaltyScore; + + return GetIntersectionScore(lhsPolygon, rhsPolygon); +} + +/// Throws NotAPolygonException exception. +/// For detailed info see comment for +/// double GetIntersectionScore(LPolygon const & lhs, RPolygon const & rhs). +/// |lhs| and |rhs| are any standard containers of m2::Point with random access iterator. +template +double GetIntersectionScoreForPoints(Container const & lhs, Container const & rhs) +{ + return GetIntersectionScore(lhs, rhs, PointsToPolygon); +} + +/// Throws NotAPolygonException exception. +/// For detailed info see comment for +/// double GetIntersectionScore(LPolygon const & lhs, RPolygon const & rhs). +/// |lhs| and |rhs| are any standard containers of m2::Point with random access iterator. +template +double GetIntersectionScoreForTriangulated(Container const & lhs, Container const & rhs) +{ + return GetIntersectionScore(lhs, rhs, TrianglesToPolygon); +} + +/// |points| is any standard container of m2::Point with random access iterator. +template +impl::Polygon PointsToPolygon(Container const & points) +{ + impl::Polygon polygon; + for (auto const & point : points) + polygon.outer().push_back(impl::PointXY(point.x, point.y)); + boost::geometry::correct(polygon); + if (!boost::geometry::is_valid(polygon)) + MYTHROW(geometry::NotAPolygonException, ("The points is not valid polygon")); + + return polygon; +} + +/// |points| is any standard container of m2::Point with random access iterator. +template +impl::MultiPolygon TrianglesToPolygon(Container const & points) +{ + size_t const kTriangleSize = 3; + if (points.size() % kTriangleSize != 0) + MYTHROW(geometry::NotAPolygonException, ("Count of points must be multiple of", kTriangleSize)); + + std::vector polygons; + for (size_t i = 0; i < points.size(); i += kTriangleSize) + { + impl::MultiPolygon polygon; + polygon.resize(1); + auto & p = polygon[0]; + auto & outer = p.outer(); + for (size_t j = i; j < i + kTriangleSize; ++j) + outer.push_back(impl::PointXY(points[j].x, points[j].y)); + boost::geometry::correct(p); + if (!boost::geometry::is_valid(polygon)) + MYTHROW(geometry::NotAPolygonException, ("The triangle is not valid")); + polygons.push_back(polygon); + } + + if (polygons.empty()) + return {}; + + auto & result = polygons[0]; + for (size_t i = 1; i < polygons.size(); ++i) + { + impl::MultiPolygon u; + boost::geometry::union_(result, polygons[i], u); + u.swap(result); + } + return result; +} +} // namespace geometry diff --git a/xcode/geometry/geometry.xcodeproj/project.pbxproj b/xcode/geometry/geometry.xcodeproj/project.pbxproj index 6cdaf891dc..71dcc55050 100644 --- a/xcode/geometry/geometry.xcodeproj/project.pbxproj +++ b/xcode/geometry/geometry.xcodeproj/project.pbxproj @@ -20,6 +20,11 @@ 39B2B9701FB4680500AB85A1 /* diamond_box.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 39B2B96E1FB4680500AB85A1 /* diamond_box.hpp */; }; 39B2B9711FB4680500AB85A1 /* diamond_box.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 39B2B96F1FB4680500AB85A1 /* diamond_box.cpp */; }; 39B2B9731FB4681400AB85A1 /* line2d.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 39B2B9721FB4681400AB85A1 /* line2d.hpp */; }; + 3D7006352463F69600AF8ADC /* intersection_score.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 3D7006342463F69600AF8ADC /* intersection_score.hpp */; }; + 3D70063A2463F6AD00AF8ADC /* polyline_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3D7006362463F6AD00AF8ADC /* polyline_tests.cpp */; }; + 3D70063B2463F6AD00AF8ADC /* circle_on_earth_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3D7006372463F6AD00AF8ADC /* circle_on_earth_tests.cpp */; }; + 3D70063C2463F6AD00AF8ADC /* area_on_earth_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3D7006382463F6AD00AF8ADC /* area_on_earth_tests.cpp */; }; + 3D70063D2463F6AD00AF8ADC /* intersection_score_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 3D7006392463F6AD00AF8ADC /* intersection_score_tests.cpp */; }; 4403990C2359CA7C0098EB7F /* segment2d_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56D9D64922A92D0A00F3D443 /* segment2d_tests.cpp */; }; 4403990D2359CA7F0098EB7F /* point3d_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 44A4BF17235732A5005857C4 /* point3d_tests.cpp */; }; 4403990E2359CA8D0098EB7F /* mercator_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = E92D8BD01C15AF9100A98D17 /* mercator_test.cpp */; }; @@ -101,10 +106,10 @@ 675344DE1A3F68F900A0A8C3 /* binary_operators.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 675344DB1A3F68F900A0A8C3 /* binary_operators.cpp */; }; 675344DF1A3F68F900A0A8C3 /* binary_operators.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 675344DC1A3F68F900A0A8C3 /* binary_operators.hpp */; }; 675344E01A3F68F900A0A8C3 /* boost_concept.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 675344DD1A3F68F900A0A8C3 /* boost_concept.hpp */; }; - D501ACA3238FDC6000B8C08E /* point_with_altitude.cpp in Sources */ = {isa = PBXBuildFile; fileRef = D501ACA1238FDC6000B8C08E /* point_with_altitude.cpp */; }; - D501ACA4238FDC6000B8C08E /* point_with_altitude.hpp in Headers */ = {isa = PBXBuildFile; fileRef = D501ACA2238FDC6000B8C08E /* point_with_altitude.hpp */; }; BBB7061323E48E0400A7F29A /* smoothing.cpp in Sources */ = {isa = PBXBuildFile; fileRef = BBB7061123E48E0400A7F29A /* smoothing.cpp */; }; BBB7061423E48E0400A7F29A /* smoothing.hpp in Headers */ = {isa = PBXBuildFile; fileRef = BBB7061223E48E0400A7F29A /* smoothing.hpp */; }; + D501ACA3238FDC6000B8C08E /* point_with_altitude.cpp in Sources */ = {isa = PBXBuildFile; fileRef = D501ACA1238FDC6000B8C08E /* point_with_altitude.cpp */; }; + D501ACA4238FDC6000B8C08E /* point_with_altitude.hpp in Headers */ = {isa = PBXBuildFile; fileRef = D501ACA2238FDC6000B8C08E /* point_with_altitude.hpp */; }; D53836352366DAF3007E7EDB /* oblate_spheroid.cpp in Sources */ = {isa = PBXBuildFile; fileRef = D53836332366DAF3007E7EDB /* oblate_spheroid.cpp */; }; D53836362366DAF3007E7EDB /* oblate_spheroid.hpp in Headers */ = {isa = PBXBuildFile; fileRef = D53836342366DAF3007E7EDB /* oblate_spheroid.hpp */; }; D53836382366DB07007E7EDB /* oblate_spheroid_tests.cpp in Sources */ = {isa = PBXBuildFile; fileRef = D53836372366DB07007E7EDB /* oblate_spheroid_tests.cpp */; }; @@ -143,6 +148,11 @@ 39B2B96E1FB4680500AB85A1 /* diamond_box.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = diamond_box.hpp; sourceTree = ""; }; 39B2B96F1FB4680500AB85A1 /* diamond_box.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = diamond_box.cpp; sourceTree = ""; }; 39B2B9721FB4681400AB85A1 /* line2d.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = line2d.hpp; sourceTree = ""; }; + 3D7006342463F69600AF8ADC /* intersection_score.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = intersection_score.hpp; sourceTree = ""; }; + 3D7006362463F6AD00AF8ADC /* polyline_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = polyline_tests.cpp; sourceTree = ""; }; + 3D7006372463F6AD00AF8ADC /* circle_on_earth_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = circle_on_earth_tests.cpp; sourceTree = ""; }; + 3D7006382463F6AD00AF8ADC /* area_on_earth_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = area_on_earth_tests.cpp; sourceTree = ""; }; + 3D7006392463F6AD00AF8ADC /* intersection_score_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = intersection_score_tests.cpp; sourceTree = ""; }; 440CF0C0236734820017C2A8 /* area_on_earth.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = area_on_earth.cpp; sourceTree = ""; }; 440CF0C1236734820017C2A8 /* area_on_earth.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = area_on_earth.hpp; sourceTree = ""; }; 4435C7872372BB5500B4358C /* circle_on_earth.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = circle_on_earth.hpp; sourceTree = ""; }; @@ -222,10 +232,10 @@ 675344DB1A3F68F900A0A8C3 /* binary_operators.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = binary_operators.cpp; sourceTree = ""; }; 675344DC1A3F68F900A0A8C3 /* binary_operators.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = binary_operators.hpp; sourceTree = ""; }; 675344DD1A3F68F900A0A8C3 /* boost_concept.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = boost_concept.hpp; sourceTree = ""; }; - D501ACA1238FDC6000B8C08E /* point_with_altitude.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = point_with_altitude.cpp; sourceTree = ""; }; - D501ACA2238FDC6000B8C08E /* point_with_altitude.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = point_with_altitude.hpp; sourceTree = ""; }; BBB7061123E48E0400A7F29A /* smoothing.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = smoothing.cpp; sourceTree = ""; }; BBB7061223E48E0400A7F29A /* smoothing.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = smoothing.hpp; sourceTree = ""; }; + D501ACA1238FDC6000B8C08E /* point_with_altitude.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = point_with_altitude.cpp; sourceTree = ""; }; + D501ACA2238FDC6000B8C08E /* point_with_altitude.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = point_with_altitude.hpp; sourceTree = ""; }; D53836332366DAF3007E7EDB /* oblate_spheroid.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = oblate_spheroid.cpp; sourceTree = ""; }; D53836342366DAF3007E7EDB /* oblate_spheroid.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = oblate_spheroid.hpp; sourceTree = ""; }; D53836372366DB07007E7EDB /* oblate_spheroid_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = oblate_spheroid_tests.cpp; sourceTree = ""; }; @@ -258,6 +268,10 @@ 670F29211BA9D25D00F2ABF4 /* geometry_tests */ = { isa = PBXGroup; children = ( + 3D7006382463F6AD00AF8ADC /* area_on_earth_tests.cpp */, + 3D7006372463F6AD00AF8ADC /* circle_on_earth_tests.cpp */, + 3D7006392463F6AD00AF8ADC /* intersection_score_tests.cpp */, + 3D7006362463F6AD00AF8ADC /* polyline_tests.cpp */, D53836372366DB07007E7EDB /* oblate_spheroid_tests.cpp */, 44A4BF17235732A5005857C4 /* point3d_tests.cpp */, 56D9D64922A92D0A00F3D443 /* segment2d_tests.cpp */, @@ -331,6 +345,7 @@ 675344931A3F684600A0A8C3 /* geometry */ = { isa = PBXGroup; children = ( + 3D7006342463F69600AF8ADC /* intersection_score.hpp */, BBB7061123E48E0400A7F29A /* smoothing.cpp */, BBB7061223E48E0400A7F29A /* smoothing.hpp */, 4435C7872372BB5500B4358C /* circle_on_earth.hpp */, @@ -436,6 +451,7 @@ 675344BF1A3F687400A0A8C3 /* avg_vector.hpp in Headers */, 675344D31A3F687400A0A8C3 /* simplification.hpp in Headers */, 675344CC1A3F687400A0A8C3 /* rect_intersect.hpp in Headers */, + 3D7006352463F69600AF8ADC /* intersection_score.hpp in Headers */, 4435C7882372BB5600B4358C /* circle_on_earth.hpp in Headers */, 675344C71A3F687400A0A8C3 /* packer.hpp in Headers */, 675344C41A3F687400A0A8C3 /* distance_on_sphere.hpp in Headers */, @@ -582,6 +598,7 @@ buildActionMask = 2147483647; files = ( 675344C31A3F687400A0A8C3 /* distance_on_sphere.cpp in Sources */, + 3D70063D2463F6AD00AF8ADC /* intersection_score_tests.cpp in Sources */, 675344CF1A3F687400A0A8C3 /* robust_orientation.cpp in Sources */, 56D5456A1C74A46D00E3719C /* algorithm.cpp in Sources */, 675344D41A3F687400A0A8C3 /* spline.cpp in Sources */, @@ -594,14 +611,17 @@ BBB7061323E48E0400A7F29A /* smoothing.cpp in Sources */, 347F33701C454205009758CC /* triangle2d.cpp in Sources */, 448425732372BB3F00C6AD04 /* circle_on_earth.cpp in Sources */, + 3D70063C2463F6AD00AF8ADC /* area_on_earth_tests.cpp in Sources */, 675344BC1A3F687400A0A8C3 /* angles.cpp in Sources */, 6741AAA01BF35476002C974C /* latlon.cpp in Sources */, E92D8BD41C15AFAA00A98D17 /* mercator.cpp in Sources */, 39B2B9711FB4680500AB85A1 /* diamond_box.cpp in Sources */, + 3D70063B2463F6AD00AF8ADC /* circle_on_earth_tests.cpp in Sources */, 675344C61A3F687400A0A8C3 /* packer.cpp in Sources */, 675344DE1A3F68F900A0A8C3 /* binary_operators.cpp in Sources */, D53836352366DAF3007E7EDB /* oblate_spheroid.cpp in Sources */, 56EB1ED01C6B6DF30022D831 /* segment2d.cpp in Sources */, + 3D70063A2463F6AD00AF8ADC /* polyline_tests.cpp in Sources */, D501ACA3238FDC6000B8C08E /* point_with_altitude.cpp in Sources */, 45A2D9C21F7526E6003310A0 /* convex_hull.cpp in Sources */, 345C55F51C93140A00B6783F /* clipping.cpp in Sources */,