[editor] Support migration for old versions of edits.

This commit is contained in:
Arsentiy Milchakov 2018-01-23 14:20:03 +03:00 committed by mgsergio
parent f80d2dbddc
commit 490416568e
3 changed files with 57 additions and 9 deletions

View file

@ -2,6 +2,7 @@
#include "base/logging.hpp"
#include "base/stl_helpers.hpp"
#include "base/stl_iterator.hpp"
#include "std/algorithm.hpp"
#include "std/function.hpp"
@ -109,8 +110,9 @@ double MatchByGeometry(LGeometry const & lhs, RGeometry const & rhs)
MultiPolygon TrianglesToPolygon(vector<m2::PointD> const & points)
{
size_t const kTriangleSize = 3;
CHECK_EQUAL(points.size() % kTriangleSize, 0, ());
CHECK(!points.empty(), ());
if (points.size() % kTriangleSize != 0 || points.empty())
MYTHROW(matcher::NotAPolygonException, ("Count of points must be multiple of", kTriangleSize));
vector<MultiPolygon> polygons;
for (size_t i = 0; i < points.size(); i += kTriangleSize)
{
@ -121,11 +123,14 @@ MultiPolygon TrianglesToPolygon(vector<m2::PointD> const & points)
for (size_t j = i; j < i + kTriangleSize; ++j)
outer.push_back(PointXY(points[j].x, points[j].y));
bg::correct(p);
ASSERT(bg::is_valid(polygon), ());
if (!bg::is_valid(polygon))
MYTHROW(matcher::NotAPolygonException, ("The triangle is not valid"));
polygons.push_back(polygon);
}
CHECK(!polygons.empty(), ());
if (polygons.empty())
return {};
auto & result = polygons[0];
for (size_t i = 1; i < polygons.size(); ++i)
{
@ -345,4 +350,28 @@ double ScoreTriangulatedGeometries(vector<m2::PointD> const & lhs, vector<m2::Po
return MatchByGeometry(lhsPolygon, rhsPolygon);
}
double ScoreTriangulatedGeometriesByPoints(vector<m2::PointD> const & lhs,
vector<m2::PointD> const & rhs)
{
// The default comparison operator used in sort above (cmp1) and one that is
// used in set_itersection (cmp2) are compatible in that sence that
// cmp2(a, b) :- cmp1(a, b) and
// cmp1(a, b) :- cmp2(a, b) || a almost equal b.
// You can think of cmp2 as !(a >= b).
// But cmp2 is not transitive:
// i.e. !cmp(a, b) && !cmp(b, c) does NOT implies !cmp(a, c),
// |a, b| < eps, |b, c| < eps.
// This could lead to unexpected results in set_itersection (with greedy implementation),
// but we assume such situation is very unlikely.
auto const matched = set_intersection(begin(lhs), end(lhs),
begin(rhs), end(rhs),
CounterIterator(),
[](m2::PointD const & p1, m2::PointD const & p2)
{
return p1 < p2 && !p1.EqualDxDy(p2, 1e-7);
}).GetCount();
return static_cast<double>(matched) / lhs.size();
}
} // namespace matcher

View file

@ -5,16 +5,24 @@
#include "geometry/mercator.hpp"
#include "geometry/point2d.hpp"
#include "base/exception.hpp"
#include "std/vector.hpp"
namespace matcher
{
DECLARE_EXCEPTION(NotAPolygonException, RootException);
/// Returns closest to the latLon node from osm or empty node if none is close enough.
pugi::xml_node GetBestOsmNode(pugi::xml_document const & osmResponse, ms::LatLon const & latLon);
/// Returns a way from osm with similar geometry or empty node if can't find such way.
/// Throws NotAPolygon exception when |geometry| is not convertible to a polygon.
pugi::xml_node GetBestOsmWayOrRelation(pugi::xml_document const & osmResponse,
vector<m2::PointD> const & geometry);
/// Returns value form [-1, 1]. Negative values are used as penalty, positive as score.
/// |lhs| and |rhs| - triangulated polygons;
/// |lhs| and |rhs| - triangulated polygons.
/// Throws NotAPolygon exception when lhs or rhs is not convertible to a polygon.
double ScoreTriangulatedGeometries(vector<m2::PointD> const & lhs, vector<m2::PointD> const & rhs);
/// Deprecated, use ScoreTriangulatedGeometries instead.
double ScoreTriangulatedGeometriesByPoints(vector<m2::PointD> const & lhs,
vector<m2::PointD> const & rhs);
} // namespace osm

View file

@ -7,7 +7,6 @@
#include "editor/feature_matcher.hpp"
#include "base/logging.hpp"
#include "base/stl_iterator.hpp"
#include "std/algorithm.hpp"
#include "std/unique_ptr.hpp"
@ -45,7 +44,7 @@ FeatureID MigrateNodeFeatureIndex(osm::Editor::ForEachFeaturesNearByFn & forEach
return feature->GetID();
}
FeatureID MigrateWayorRelatonFeatureIndex(
FeatureID MigrateWayOrRelatonFeatureIndex(
osm::Editor::ForEachFeaturesNearByFn & forEach, XMLFeature const & xml,
osm::Editor::FeatureStatus const /* Unused for now (we don't create/delete area features)*/,
TGenerateIDFn const & /*Unused for the same reason*/)
@ -69,7 +68,19 @@ FeatureID MigrateWayorRelatonFeatureIndex(
++count;
auto ftGeometry = ft.GetTriangesAsPoints(FeatureType::BEST_GEOMETRY);
auto const score = matcher::ScoreTriangulatedGeometries(geometry, ftGeometry);
double score = 0.0;
try
{
score = matcher::ScoreTriangulatedGeometries(geometry, ftGeometry);
}
catch (matcher::NotAPolygonException & ex)
{
// Support migration for old application versions.
// TODO(a): To remove it after some time.
my::SortUnique(geometry);
my::SortUnique(ftGeometry);
score = matcher::ScoreTriangulatedGeometriesByPoints(geometry, ftGeometry);
}
if (score > bestScore)
{
@ -103,7 +114,7 @@ FeatureID MigrateFeatureIndex(osm::Editor::ForEachFeaturesNearByFn & forEach,
return MigrateNodeFeatureIndex(forEach, xml, featureStatus, generateID);
case XMLFeature::Type::Way:
case XMLFeature::Type::Relation:
return MigrateWayorRelatonFeatureIndex(forEach, xml, featureStatus, generateID);
return MigrateWayOrRelatonFeatureIndex(forEach, xml, featureStatus, generateID);
}
}
} // namespace editor