Fix feature generation for boundary-administrative-* types. Do not assign such types for inner polygons inside border relation.

This commit is contained in:
vng 2012-04-20 14:07:17 +03:00 committed by Alex Zolotarev
parent 567a594e8c
commit 5841f77fd2
2 changed files with 59 additions and 16 deletions

View file

@ -99,6 +99,13 @@ struct RelationElement
for (size_t i = 0; i < ways.size(); ++i)
toDo(ways[i].first, ways[i].second);
}
void Swap(RelationElement & rhs)
{
nodes.swap(rhs.nodes);
ways.swap(rhs.ways);
tags.swap(rhs.tags);
}
};
template <class TArchive> TArchive & operator << (TArchive & ar, WayElement const & e)

View file

@ -207,9 +207,34 @@ protected:
FeatureParams * m_val;
/// Cache: relation id -> feature value (for fast feature parsing)
unordered_map<uint64_t, FeatureParams> m_typeCache;
struct RelationValue
{
FeatureParams m_p;
RelationElement * m_e;
RelationValue() : m_e(0) {}
};
typedef unordered_map<uint64_t, RelationValue> RelationCacheT;
RelationCacheT m_typeCache;
bool IsAcceptTypes(RelationElement const & rel) const
{
string role;
VERIFY ( rel.FindWay(m_featureID, role), (id, m_featureID) );
// Do not accumulate border types (boundary-administrative-*) for inner polygons.
// Example: Minsk city border (admin_level=8) is inner for Minsk area border (admin_level=4).
return (role != "inner");
}
public:
~type_processor()
{
for (typename RelationCacheT::iterator i = m_typeCache.begin(); i != m_typeCache.end(); ++i)
delete i->second.m_e;
}
/// Start process new feature.
void Reset(uint64_t fID, FeatureParams * val)
{
@ -220,39 +245,50 @@ protected:
/// 1. "initial relation" process
int operator() (uint64_t id)
{
typename unordered_map<uint64_t, FeatureParams>::const_iterator i = m_typeCache.find(id);
typename RelationCacheT::const_iterator i = m_typeCache.find(id);
if (i != m_typeCache.end())
{
m_val->AddTypes(i->second);
if (i->second.m_e == 0 || IsAcceptTypes(*(i->second.m_e)))
m_val->AddTypes(i->second.m_p);
return -1; // continue process relations
}
return 0; // read relation from file (see next operator)
}
/// 2. "relation from file" process
bool operator() (uint64_t id, RelationElement const & rel)
/// param[in] rel Get non-const reference to Swap inner data
bool operator() (uint64_t id, RelationElement & rel)
{
string const type = rel.GetType();
if (type == "multipolygon")
{
// we will process multipolygons later
return false;
}
bool const isBoundary = (type == "boundary");
// make XMLElement struct from relation's tags for GetNameAndType function.
XMLElement e;
make_xml_element(rel, e);
if (rel.GetType() == "multipolygon")
// process types of relation and add them to m_val
RelationValue val;
ftype::GetNameAndType(&e, val.m_p);
if (val.m_p.IsValid())
{
// we will process multipolygons later
}
else
{
// process types of relation and add them to m_val
FeatureParams val;
if (ftype::GetNameAndType(&e, val))
if (!isBoundary || IsAcceptTypes(rel))
m_val->AddTypes(val.m_p);
if (isBoundary)
{
m_typeCache[id] = val;
m_val->AddTypes(val);
val.m_e = new RelationElement();
val.m_e->Swap(rel);
}
else
m_typeCache[id] = FeatureParams();
}
m_typeCache[id] = val;
// continue process relations
return false;
}