From 61711f54b57a7fded06f9454ecf412c6976cc4ad Mon Sep 17 00:00:00 2001 From: Mikhail Gorbushin Date: Fri, 11 Oct 2019 14:46:32 +0300 Subject: [PATCH] [geometry] move AreaOnEarth functions to area_on_earth.* files --- geometry/CMakeLists.txt | 2 + geometry/area_on_earth.cpp | 66 +++++++++++++++++++ geometry/area_on_earth.hpp | 18 +++++ geometry/distance_on_sphere.cpp | 51 -------------- geometry/distance_on_sphere.hpp | 11 ---- geometry/geometry_tests/CMakeLists.txt | 1 + .../geometry_tests/area_on_earth_tests.cpp | 38 +++++++++++ .../distance_on_sphere_test.cpp | 35 ---------- geometry/mercator.cpp | 1 + .../geometry.xcodeproj/project.pbxproj | 8 +++ 10 files changed, 134 insertions(+), 97 deletions(-) create mode 100644 geometry/area_on_earth.cpp create mode 100644 geometry/area_on_earth.hpp create mode 100644 geometry/geometry_tests/area_on_earth_tests.cpp diff --git a/geometry/CMakeLists.txt b/geometry/CMakeLists.txt index 4a583680f6..3ddcd864e5 100644 --- a/geometry/CMakeLists.txt +++ b/geometry/CMakeLists.txt @@ -7,6 +7,8 @@ set( angles.cpp angles.hpp any_rect2d.hpp + area_on_earth.cpp + area_on_earth.hpp avg_vector.hpp bounding_box.cpp bounding_box.hpp diff --git a/geometry/area_on_earth.cpp b/geometry/area_on_earth.cpp new file mode 100644 index 0000000000..4d724d792c --- /dev/null +++ b/geometry/area_on_earth.cpp @@ -0,0 +1,66 @@ +#include "geometry/area_on_earth.hpp" + +#include "geometry/convex_hull.hpp" +#include "geometry/distance_on_sphere.hpp" +#include "geometry/mercator.hpp" +#include "geometry/point3d.hpp" +#include "geometry/polygon.hpp" +#include "geometry/segment2d.hpp" + +#include "base/assert.hpp" + +#include +#include + +namespace +{ +double const kEarthRadiusMetersSquared = ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; +} // namespace + +namespace ms +{ +// Look to https://en.wikipedia.org/wiki/Solid_angle for more details. +// Shortly: +// It's possible to calculate area of triangle on sphere with it's solid angle. +// Ω = A / R^2 +// Where Ω is solid angle of triangle, R - sphere radius and A - area of triangle on sphere. +// So A = Ω * R^2 +double AreaOnEarth(LatLon const & ll1, LatLon const & ll2, LatLon const & ll3) +{ + m3::PointD const a = GetPointOnSphere(ll1, 1.0 /* sphereRadius */); + m3::PointD const b = GetPointOnSphere(ll2, 1.0 /* sphereRadius */); + m3::PointD const c = GetPointOnSphere(ll3, 1.0 /* sphereRadius */); + + double const triple = m3::DotProduct(a, m3::CrossProduct(b, c)); + + ASSERT(base::AlmostEqualAbs(a.Length(), 1.0, 1e-5), ()); + ASSERT(base::AlmostEqualAbs(b.Length(), 1.0, 1e-5), ()); + ASSERT(base::AlmostEqualAbs(c.Length(), 1.0, 1e-5), ()); + + double constexpr lengthMultiplication = 1.0; // a.Length() * b.Length() * c.Length() + double const abc = m3::DotProduct(a, b); // * c.Length() == 1 + double const acb = m3::DotProduct(a, c); // * b.Length() == 1 + double const bca = m3::DotProduct(b, c); // * a.Length() == 1 + + double const tanFromHalfSolidAngle = triple / (lengthMultiplication + abc + acb + bca); + double const halfSolidAngle = atan(tanFromHalfSolidAngle); + double const solidAngle = halfSolidAngle * 2.0; + double const area = solidAngle * kEarthRadiusMetersSquared; + return abs(area); +} + +// Look to https://en.wikipedia.org/wiki/Solid_angle for details. +// Shortly: +// Ω = A / R^2 +// A is the spherical surface area which confined by solid angle. +// R - sphere radius. +// For circle: Ω = 2π(1 - cos(θ / 2)), where θ - is the cone apex angle. +double CircleAreaOnEarth(double distanceOnSphereRadius) +{ + double const theta = 2.0 * distanceOnSphereRadius / ms::kEarthRadiusMeters; + static double const kConst = 2 * math::pi * kEarthRadiusMetersSquared; + double const sinValue = sin(theta / 4); + // 1 - cos(θ / 2) = 2sin^2(θ / 4) + return kConst * 2 * sinValue * sinValue; +} +} // namespace ms diff --git a/geometry/area_on_earth.hpp b/geometry/area_on_earth.hpp new file mode 100644 index 0000000000..b26301974c --- /dev/null +++ b/geometry/area_on_earth.hpp @@ -0,0 +1,18 @@ +#pragma once + +#include "geometry/latlon.hpp" +#include "geometry/point2d.hpp" + +namespace ms +{ +// Returns area of triangle on earth. +double AreaOnEarth(LatLon const & ll1, LatLon const & ll2, LatLon const & ll3); + +// Area of the spherical cap that contains all points +// within the distance |radius| from an arbitrary fixed point, measured +// along the Earth surface. +// In particular, the smallest cap spanning the whole Earth results +// from radius = pi*EarthRadius. +// For small enough radiuses, returns the value close to pi*|radius|^2. +double CircleAreaOnEarth(double radius); +} // namespace ms diff --git a/geometry/distance_on_sphere.cpp b/geometry/distance_on_sphere.cpp index c140c0a2a7..a72da80ca1 100644 --- a/geometry/distance_on_sphere.cpp +++ b/geometry/distance_on_sphere.cpp @@ -10,11 +10,6 @@ using namespace std; -namespace -{ -double const kEarthRadiusMetersSquared = ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; -} // namespace - namespace ms { double DistanceOnSphere(double lat1Deg, double lon1Deg, double lat2Deg, double lon2Deg) @@ -51,50 +46,4 @@ m3::PointD GetPointOnSphere(ms::LatLon const & ll, double sphereRadius) return {x, y, z}; } - -// Look to https://en.wikipedia.org/wiki/Solid_angle for more details. -// Shortly: -// It's possible to calculate area of triangle on sphere with it's solid angle. -// Ω = A / R^2 -// Where Ω is solid angle of triangle, R - sphere radius and A - area of triangle on sphere. -// So A = Ω * R^2 -double AreaOnEarth(LatLon const & ll1, LatLon const & ll2, LatLon const & ll3) -{ - m3::PointD const a = GetPointOnSphere(ll1, 1.0 /* sphereRadius */); - m3::PointD const b = GetPointOnSphere(ll2, 1.0 /* sphereRadius */); - m3::PointD const c = GetPointOnSphere(ll3, 1.0 /* sphereRadius */); - - double const triple = m3::DotProduct(a, m3::CrossProduct(b, c)); - - ASSERT(base::AlmostEqualAbs(a.Length(), 1.0, 1e-5), ()); - ASSERT(base::AlmostEqualAbs(b.Length(), 1.0, 1e-5), ()); - ASSERT(base::AlmostEqualAbs(c.Length(), 1.0, 1e-5), ()); - - double constexpr lengthMultiplication = 1.0; // a.Length() * b.Length() * c.Length() - double const abc = m3::DotProduct(a, b); // * c.Length() == 1 - double const acb = m3::DotProduct(a, c); // * b.Length() == 1 - double const bca = m3::DotProduct(b, c); // * a.Length() == 1 - - double const tanFromHalfSolidAngle = triple / (lengthMultiplication + abc + acb + bca); - double const halfSolidAngle = atan(tanFromHalfSolidAngle); - double const solidAngle = halfSolidAngle * 2.0; - double const area = solidAngle * kEarthRadiusMetersSquared; - return abs(area); -} - -// Look to https://en.wikipedia.org/wiki/Solid_angle for details. -// Shortly: -// Ω = A / R^2 -// A is the spherical surface area which confined by solid angle. -// R - sphere radius. -// For circle: Ω = 2π(1 - cos(θ / 2)), where θ - is the cone apex angle. -double CircleAreaOnEarth(double radius) -{ - radius = base::Clamp(radius, 0.0, math::pi * ms::kEarthRadiusMeters); - double const theta = 2.0 * radius / ms::kEarthRadiusMeters; - static double const kConst = 2.0 * math::pi * kEarthRadiusMetersSquared; - double const sinValue = sin(theta / 4.0); - // 1 - cos(θ / 2) = 2sin^2(θ / 4) - return kConst * 2.0 * sinValue * sinValue; -} } // namespace ms diff --git a/geometry/distance_on_sphere.hpp b/geometry/distance_on_sphere.hpp index 4996c70744..93f2bea9f4 100644 --- a/geometry/distance_on_sphere.hpp +++ b/geometry/distance_on_sphere.hpp @@ -19,15 +19,4 @@ double DistanceOnEarth(double lat1Deg, double lon1Deg, double lat2Deg, double lo double DistanceOnEarth(LatLon const & ll1, LatLon const & ll2); m3::PointD GetPointOnSphere(LatLon const & ll, double sphereRadius); - -// Returns area of triangle on earth. -double AreaOnEarth(LatLon const & ll1, LatLon const & ll2, LatLon const & ll3); - -// Area of the spherical cap that contains all points -// within the distance |radius| from an arbitrary fixed point, measured -// along the Earth surface. -// In particular, the smallest cap spanning the whole Earth results -// from radius = pi*EarthRadius. -// For small enough radiuses, returns the value close to pi*|radius|^2. -double CircleAreaOnEarth(double radius); } // namespace ms diff --git a/geometry/geometry_tests/CMakeLists.txt b/geometry/geometry_tests/CMakeLists.txt index 57dee9ed67..6e7f43ad2e 100644 --- a/geometry/geometry_tests/CMakeLists.txt +++ b/geometry/geometry_tests/CMakeLists.txt @@ -7,6 +7,7 @@ set( algorithm_test.cpp angle_test.cpp anyrect_test.cpp + area_on_earth_tests.cpp bounding_box_tests.cpp calipers_box_tests.cpp cellid_test.cpp diff --git a/geometry/geometry_tests/area_on_earth_tests.cpp b/geometry/geometry_tests/area_on_earth_tests.cpp new file mode 100644 index 0000000000..5f149860b1 --- /dev/null +++ b/geometry/geometry_tests/area_on_earth_tests.cpp @@ -0,0 +1,38 @@ +#include "testing/testing.hpp" + +#include "geometry/area_on_earth.hpp" +#include "geometry/distance_on_sphere.hpp" +#include "geometry/mercator.hpp" + +#include "base/math.hpp" + +#include + +UNIT_TEST(AreaOnEarth_Circle) +{ + double const kEarthSurfaceArea = 4.0 * math::pi * ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; + TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(math::pi * ms::kEarthRadiusMeters), + kEarthSurfaceArea, + 1e-1, ()); + + TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(2.0 /* radiusMeters */), + math::pi * 2.0 * 2.0, + 1e-2, ()); + + TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(2000.0 /* radiusMeters */), + math::pi * 2000.0 * 2000.0, + 1.0, ()); +} + +UNIT_TEST(AreaOnEarth_ThreePoints) +{ + double const kEarthSurfaceArea = 4.0 * math::pi * ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; + + TEST_ALMOST_EQUAL_ABS(ms::AreaOnEarth({90.0, 0.0}, {0.0, 0.0}, {0.0, 90.0}), + kEarthSurfaceArea / 8.0, + 1e-2, ()); + + TEST_ALMOST_EQUAL_ABS(ms::AreaOnEarth({90.0, 0.0}, {0.0, 90.0}, {0.0, -90.0}), + kEarthSurfaceArea / 4.0, + 1e-1, ()); +} \ No newline at end of file diff --git a/geometry/geometry_tests/distance_on_sphere_test.cpp b/geometry/geometry_tests/distance_on_sphere_test.cpp index 2aac586a4c..6adf2e1b90 100644 --- a/geometry/geometry_tests/distance_on_sphere_test.cpp +++ b/geometry/geometry_tests/distance_on_sphere_test.cpp @@ -22,38 +22,3 @@ UNIT_TEST(DistanceOnEarth) TEST_LESS(fabs(ms::DistanceOnEarth(47.37, 8.56, 53.91, 27.56) * 0.001 - 1519), 1, ()); TEST_LESS(fabs(ms::DistanceOnEarth(43, 132, 38, -122.5) * 0.001 - 8302), 1, ()); } - -UNIT_TEST(CircleAreaOnEarth) -{ - double const kEarthSurfaceArea = 4.0 * math::pi * ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; - TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(math::pi * ms::kEarthRadiusMeters), - kEarthSurfaceArea, - 1e-1, ()); - - TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(2.0 /* radiusMeters */), - math::pi * 2.0 * 2.0, - 1e-2, ()); - - TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(2000.0 /* radiusMeters */), - math::pi * 2000.0 * 2000.0, - 1.0, ()); - - double constexpr kVeryBigRadius = 1e100; - CHECK_GREATER(kVeryBigRadius, ms::kEarthRadiusMeters, ()); - TEST_ALMOST_EQUAL_ABS(ms::CircleAreaOnEarth(kVeryBigRadius), - kEarthSurfaceArea, - 1e-1, ()); -} - -UNIT_TEST(AreaOnEarth_ThreePoints) -{ - double const kEarthSurfaceArea = 4.0 * math::pi * ms::kEarthRadiusMeters * ms::kEarthRadiusMeters; - - TEST_ALMOST_EQUAL_ABS(ms::AreaOnEarth({90.0, 0.0}, {0.0, 0.0}, {0.0, 90.0}), - kEarthSurfaceArea / 8.0, - 1e-2, ()); - - TEST_ALMOST_EQUAL_ABS(ms::AreaOnEarth({90.0, 0.0}, {0.0, 90.0}, {0.0, -90.0}), - kEarthSurfaceArea / 4.0, - 1e-1, ()); -} diff --git a/geometry/mercator.cpp b/geometry/mercator.cpp index f1a9e46258..51d122d9b3 100644 --- a/geometry/mercator.cpp +++ b/geometry/mercator.cpp @@ -1,5 +1,6 @@ #include "geometry/mercator.hpp" +#include "geometry/area_on_earth.hpp" #include "geometry/distance_on_sphere.hpp" #include diff --git a/xcode/geometry/geometry.xcodeproj/project.pbxproj b/xcode/geometry/geometry.xcodeproj/project.pbxproj index 210c9fb2dc..26eab53446 100644 --- a/xcode/geometry/geometry.xcodeproj/project.pbxproj +++ b/xcode/geometry/geometry.xcodeproj/project.pbxproj @@ -23,6 +23,8 @@ 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 */; }; + 440CF0C2236734820017C2A8 /* area_on_earth.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 440CF0C0236734820017C2A8 /* area_on_earth.cpp */; }; + 440CF0C3236734820017C2A8 /* area_on_earth.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 440CF0C1236734820017C2A8 /* area_on_earth.hpp */; }; 446531E2234DEEF3000C348F /* point3d.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 446531E1234DEEF3000C348F /* point3d.hpp */; }; 45A2D9BE1F7526E6003310A0 /* bounding_box.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 45A2D9B71F7526E6003310A0 /* bounding_box.cpp */; }; 45A2D9BF1F7526E6003310A0 /* bounding_box.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 45A2D9B81F7526E6003310A0 /* bounding_box.hpp */; }; @@ -135,6 +137,8 @@ 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 = ""; }; + 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 = ""; }; 446531E1234DEEF3000C348F /* point3d.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = point3d.hpp; sourceTree = ""; }; 44A4BF17235732A5005857C4 /* point3d_tests.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = point3d_tests.cpp; sourceTree = ""; }; 45A2D9B71F7526E6003310A0 /* bounding_box.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = bounding_box.cpp; sourceTree = ""; }; @@ -317,6 +321,8 @@ children = ( D53836332366DAF3007E7EDB /* oblate_spheroid.cpp */, D53836342366DAF3007E7EDB /* oblate_spheroid.hpp */, + 440CF0C0236734820017C2A8 /* area_on_earth.cpp */, + 440CF0C1236734820017C2A8 /* area_on_earth.hpp */, 56D545681C74A46D00E3719C /* algorithm.cpp */, 56D545691C74A46D00E3719C /* algorithm.hpp */, 6753449F1A3F687400A0A8C3 /* angles.cpp */, @@ -413,6 +419,7 @@ 675344CC1A3F687400A0A8C3 /* rect_intersect.hpp in Headers */, 675344C71A3F687400A0A8C3 /* packer.hpp in Headers */, 675344C41A3F687400A0A8C3 /* distance_on_sphere.hpp in Headers */, + 440CF0C3236734820017C2A8 /* area_on_earth.hpp in Headers */, 45A2D9C11F7526E6003310A0 /* calipers_box.hpp in Headers */, 675344CA1A3F687400A0A8C3 /* polygon.hpp in Headers */, 675344C11A3F687400A0A8C3 /* covering_utils.hpp in Headers */, @@ -559,6 +566,7 @@ 675344D41A3F687400A0A8C3 /* spline.cpp in Sources */, 45A2D9C41F7526E6003310A0 /* line2d.cpp in Sources */, 675344D11A3F687400A0A8C3 /* screenbase.cpp in Sources */, + 440CF0C2236734820017C2A8 /* area_on_earth.cpp in Sources */, 45A2D9BE1F7526E6003310A0 /* bounding_box.cpp in Sources */, 45A2D9C01F7526E6003310A0 /* calipers_box.cpp in Sources */, 344A713C1F3DA07000B8DDB8 /* nearby_points_sweeper.cpp in Sources */,