forked from organicmaps/organicmaps
Moving GraphData to routing_common.
This commit is contained in:
parent
2d54674e2a
commit
0cb28f0e4d
8 changed files with 771 additions and 694 deletions
|
@ -4,43 +4,34 @@
|
|||
#include "generator/borders_loader.hpp"
|
||||
#include "generator/utils.hpp"
|
||||
|
||||
#include "traffic/traffic_cache.hpp"
|
||||
|
||||
#include "routing/routing_exceptions.hpp"
|
||||
#include "routing/index_router.hpp"
|
||||
#include "routing/routing_exceptions.hpp"
|
||||
#include "routing/vehicle_mask.hpp"
|
||||
|
||||
#include "routing_common/transit_serdes.hpp"
|
||||
#include "routing_common/transit_speed_limits.hpp"
|
||||
#include "routing_common/transit_types.hpp"
|
||||
|
||||
#include "storage/country_info_getter.hpp"
|
||||
#include "storage/routing_helpers.hpp"
|
||||
|
||||
#include "indexer/index.hpp"
|
||||
#include "indexer/mwm_set.hpp"
|
||||
|
||||
#include "geometry/mercator.hpp"
|
||||
#include "geometry/point2d.hpp"
|
||||
#include "geometry/rect2d.hpp"
|
||||
#include "geometry/region2d.hpp"
|
||||
|
||||
#include "coding/file_container.hpp"
|
||||
#include "coding/file_name_utils.hpp"
|
||||
#include "coding/file_writer.hpp"
|
||||
|
||||
#include "platform/country_file.hpp"
|
||||
#include "platform/local_country_file.hpp"
|
||||
#include "platform/local_country_file_utils.hpp"
|
||||
#include "platform/platform.hpp"
|
||||
|
||||
#include "base/assert.hpp"
|
||||
#include "base/checked_cast.hpp"
|
||||
#include "base/exception.hpp"
|
||||
#include "base/logging.hpp"
|
||||
#include "base/macros.hpp"
|
||||
#include "base/stl_helpers.hpp"
|
||||
#include "base/string_utils.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
#include <functional>
|
||||
#include <iterator>
|
||||
#include <memory>
|
||||
#include <utility>
|
||||
#include <vector>
|
||||
|
||||
#include "defines.hpp"
|
||||
|
||||
#include "3party/jansson/src/jansson.h"
|
||||
|
||||
|
@ -53,69 +44,6 @@ using namespace storage;
|
|||
|
||||
namespace
|
||||
{
|
||||
struct ClearVisitor
|
||||
{
|
||||
template <typename Cont>
|
||||
void operator()(Cont & c, char const * /* name */) const { c.clear(); }
|
||||
};
|
||||
|
||||
struct SortVisitor
|
||||
{
|
||||
template <typename Cont>
|
||||
void operator()(Cont & c, char const * /* name */) const
|
||||
{
|
||||
sort(c.begin(), c.end());
|
||||
}
|
||||
};
|
||||
|
||||
struct IsValidVisitor
|
||||
{
|
||||
template <typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */ )
|
||||
{
|
||||
m_isValid = m_isValid && ::IsValid(c);
|
||||
}
|
||||
|
||||
bool IsValid() const { return m_isValid; }
|
||||
|
||||
private:
|
||||
bool m_isValid = true;
|
||||
};
|
||||
|
||||
struct IsUniqueVisitor
|
||||
{
|
||||
template <typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */)
|
||||
{
|
||||
m_isUnique = m_isUnique && (adjacent_find(c.cbegin(), c.cend()) == c.cend());
|
||||
}
|
||||
|
||||
bool IsUnique() const { return m_isUnique; }
|
||||
|
||||
private:
|
||||
bool m_isUnique = true;
|
||||
};
|
||||
|
||||
struct IsSortedVisitor
|
||||
{
|
||||
template <typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */)
|
||||
{
|
||||
m_isSorted = m_isSorted && is_sorted(c.cbegin(), c.cend());
|
||||
}
|
||||
|
||||
bool IsSorted() const { return m_isSorted; }
|
||||
|
||||
private:
|
||||
bool m_isSorted = true;
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
void Append(vector<T> const & src, vector<T> & dst)
|
||||
{
|
||||
dst.insert(dst.end(), src.begin(), src.end());
|
||||
}
|
||||
|
||||
void LoadBorders(string const & dir, TCountryId const & countryId, vector<m2::RegionD> & borders)
|
||||
{
|
||||
string const polyFile = my::JoinPath(dir, BORDERS_DIR, countryId + BORDERS_EXTENSION);
|
||||
|
@ -123,58 +51,6 @@ void LoadBorders(string const & dir, TCountryId const & countryId, vector<m2::Re
|
|||
osm::LoadBorders(polyFile, borders);
|
||||
}
|
||||
|
||||
bool HasStop(vector<Stop> const & stops, StopId stopId)
|
||||
{
|
||||
return binary_search(stops.cbegin(), stops.cend(), Stop(stopId));
|
||||
}
|
||||
|
||||
/// \brief Removes from |items| all items which stop ids is not contained in |stops|.
|
||||
/// \note This method keeps relative order of |items|.
|
||||
template <class Item>
|
||||
void ClipItemsByStops(vector<Stop> const & stops, vector<Item> & items)
|
||||
{
|
||||
CHECK(is_sorted(stops.cbegin(), stops.cend()), ());
|
||||
|
||||
vector<Item> itemsToFill;
|
||||
for (auto const & item : items)
|
||||
{
|
||||
for (auto const stopId : item.GetStopIds())
|
||||
{
|
||||
if (HasStop(stops, stopId))
|
||||
{
|
||||
itemsToFill.push_back(item);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
items.swap(itemsToFill);
|
||||
}
|
||||
|
||||
/// \returns ref to an item at |items| by |id|.
|
||||
/// \note |items| must be sorted before a call of this method.
|
||||
template <class Id, class Item>
|
||||
Item const & FindById(vector<Item> const & items, Id id)
|
||||
{
|
||||
auto const s1Id = equal_range(items.cbegin(), items.cend(), Item(id));
|
||||
CHECK_EQUAL(distance(s1Id.first, s1Id.second), 1,
|
||||
("An item with id:", id, "is not unique or there's not such item. items:", items));
|
||||
return *s1Id.first;
|
||||
}
|
||||
|
||||
/// \brief Fills |items| with items which have ids from |ids|.
|
||||
/// \note |items| must be sorted before a call of this method.
|
||||
template <class Id, class Item>
|
||||
void UpdateItems(set<Id> const & ids, vector<Item> & items)
|
||||
{
|
||||
vector<Item> itemsToFill;
|
||||
for (auto const id : ids)
|
||||
itemsToFill.push_back(FindById(items, id));
|
||||
|
||||
SortVisitor{}(itemsToFill, nullptr /* name */);
|
||||
items.swap(itemsToFill);
|
||||
}
|
||||
|
||||
void FillOsmIdToFeatureIdsMap(string const & osmIdToFeatureIdsPath, OsmIdToFeatureIdsMap & map)
|
||||
{
|
||||
CHECK(ForEachOsmId2FeatureId(osmIdToFeatureIdsPath,
|
||||
|
@ -189,235 +65,10 @@ string GetMwmPath(string const & mwmDir, TCountryId const & countryId)
|
|||
return my::JoinPath(mwmDir, countryId + DATA_FILE_EXTENSION);
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
namespace routing
|
||||
{
|
||||
namespace transit
|
||||
{
|
||||
// DeserializerFromJson ---------------------------------------------------------------------------
|
||||
DeserializerFromJson::DeserializerFromJson(json_struct_t * node,
|
||||
OsmIdToFeatureIdsMap const & osmIdToFeatureIds)
|
||||
: m_node(node), m_osmIdToFeatureIds(osmIdToFeatureIds)
|
||||
{
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(m2::PointD & p, char const * name)
|
||||
{
|
||||
// @todo(bykoianko) Instead of having a special operator() method for m2::PointD class it's necessary to
|
||||
// add Point class to transit_types.hpp and process it in DeserializerFromJson with regular method.
|
||||
json_t * item = nullptr;
|
||||
if (name == nullptr)
|
||||
item = m_node; // Array item case
|
||||
else
|
||||
item = my::GetJSONObligatoryField(m_node, name);
|
||||
|
||||
CHECK(json_is_object(item), ("Item is not a json object:", name));
|
||||
FromJSONObject(item, "x", p.x);
|
||||
FromJSONObject(item, "y", p.y);
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(FeatureIdentifiers & id, char const * name)
|
||||
{
|
||||
// Conversion osm id to feature id.
|
||||
string osmIdStr;
|
||||
GetField(osmIdStr, name);
|
||||
CHECK(strings::is_number(osmIdStr), ("Osm id string is not a number:", osmIdStr));
|
||||
uint64_t osmIdNum;
|
||||
CHECK(strings::to_uint64(osmIdStr, osmIdNum),
|
||||
("Cann't convert osm id string:", osmIdStr, "to a number."));
|
||||
osm::Id const osmId(osmIdNum);
|
||||
auto const it = m_osmIdToFeatureIds.find(osmId);
|
||||
if (it != m_osmIdToFeatureIds.cend())
|
||||
{
|
||||
CHECK_GREATER_OR_EQUAL(it->second.size(), 1, ("Osm id:", osmId, "(encoded", osmId.EncodedId(),
|
||||
") from transit graph does not correspond to any feature."));
|
||||
if (it->second.size() != 1)
|
||||
{
|
||||
// Note. |osmId| corresponds several feature ids. It may happen in case of stops;
|
||||
// if a stop is present as relation. It's a rare case.
|
||||
LOG(LWARNING, ("Osm id:", osmId, "(encoded", osmId.EncodedId(), "corresponds to",
|
||||
it->second.size(), "feature ids."));
|
||||
}
|
||||
id.SetFeatureId(it->second[0]);
|
||||
}
|
||||
id.SetOsmId(osmId.EncodedId());
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(EdgeFlags & edgeFlags, char const * name)
|
||||
{
|
||||
bool transfer = false;
|
||||
(*this)(transfer, name);
|
||||
// Note. Only |transfer| field of |edgeFlags| may be set at this point because the
|
||||
// other fields of |edgeFlags| are unknown.
|
||||
edgeFlags.SetFlags(0);
|
||||
edgeFlags.m_transfer = transfer;
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(StopIdRanges & rs, char const * name)
|
||||
{
|
||||
vector<StopId> stopIds;
|
||||
(*this)(stopIds, name);
|
||||
rs = StopIdRanges({stopIds});
|
||||
}
|
||||
|
||||
// GraphData --------------------------------------------------------------------------------------
|
||||
void GraphData::DeserializeFromJson(my::Json const & root, OsmIdToFeatureIdsMap const & mapping)
|
||||
{
|
||||
DeserializerFromJson deserializer(root.get(), mapping);
|
||||
Visit(deserializer);
|
||||
|
||||
// Removes equivalent edges from |m_edges|. If there are several equivalent edges only
|
||||
// the most lightweight edge is left.
|
||||
// Note. It's possible that two stops are connected with the same line several times
|
||||
// in the same direction. It happens in Oslo metro (T-banen):
|
||||
// https://en.wikipedia.org/wiki/Oslo_Metro#/media/File:Oslo_Metro_Map.svg branch 5.
|
||||
my::SortUnique(m_edges,
|
||||
[](Edge const & e1, Edge const & e2) {
|
||||
if (e1 != e2)
|
||||
return e1 < e2;
|
||||
return e1.GetWeight() < e2.GetWeight();
|
||||
},
|
||||
[](Edge const & e1, Edge const & e2) { return e1 == e2; });
|
||||
}
|
||||
|
||||
void GraphData::Serialize(Writer & writer) const
|
||||
{
|
||||
TransitHeader header;
|
||||
|
||||
auto const startOffset = writer.Pos();
|
||||
Serializer<Writer> serializer(writer);
|
||||
FixedSizeSerializer<Writer> numberSerializer(writer);
|
||||
numberSerializer(header);
|
||||
header.m_stopsOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_stops);
|
||||
header.m_gatesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_gates);
|
||||
header.m_edgesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_edges);
|
||||
header.m_transfersOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_transfers);
|
||||
header.m_linesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_lines);
|
||||
header.m_shapesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_shapes);
|
||||
header.m_networksOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_networks);
|
||||
header.m_endOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
// Rewriting header info.
|
||||
CHECK(header.IsValid(), (header));
|
||||
auto const endOffset = writer.Pos();
|
||||
writer.Seek(startOffset);
|
||||
numberSerializer(header);
|
||||
writer.Seek(endOffset);
|
||||
|
||||
LOG(LINFO, (TRANSIT_FILE_TAG, "section is ready. Header:", header));
|
||||
}
|
||||
|
||||
void GraphData::Deserialize(Reader & reader)
|
||||
{
|
||||
NonOwningReaderSource src(reader);
|
||||
transit::Deserializer<NonOwningReaderSource> deserializer(src);
|
||||
transit::FixedSizeDeserializer<NonOwningReaderSource> numberDeserializer(src);
|
||||
|
||||
transit::TransitHeader header;
|
||||
numberDeserializer(header);
|
||||
CHECK(header.IsValid(), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_stopsOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_stops);
|
||||
CHECK(transit::IsValidSortedUnique(m_stops), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_gatesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_gates);
|
||||
CHECK(transit::IsValidSortedUnique(m_gates), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_edgesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_edges);
|
||||
CHECK(transit::IsValidSortedUnique(m_edges), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_transfersOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_transfers);
|
||||
CHECK(transit::IsValidSortedUnique(m_transfers), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_linesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_lines);
|
||||
CHECK(transit::IsValidSortedUnique(m_lines), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_shapesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_shapes);
|
||||
CHECK(transit::IsValidSortedUnique(m_shapes), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_networksOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_networks);
|
||||
CHECK(transit::IsValidSortedUnique(m_networks), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_endOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
}
|
||||
|
||||
void GraphData::AppendTo(GraphData const & rhs)
|
||||
{
|
||||
::Append(rhs.m_stops, m_stops);
|
||||
::Append(rhs.m_gates, m_gates);
|
||||
::Append(rhs.m_edges, m_edges);
|
||||
::Append(rhs.m_transfers, m_transfers);
|
||||
::Append(rhs.m_lines, m_lines);
|
||||
::Append(rhs.m_shapes, m_shapes);
|
||||
::Append(rhs.m_networks, m_networks);
|
||||
}
|
||||
|
||||
void GraphData::Clear()
|
||||
{
|
||||
ClearVisitor const v{};
|
||||
Visit(v);
|
||||
}
|
||||
|
||||
bool GraphData::IsValid() const
|
||||
{
|
||||
if (!IsSorted() || !IsUnique())
|
||||
return false;
|
||||
|
||||
IsValidVisitor v;
|
||||
Visit(v);
|
||||
return v.IsValid();
|
||||
}
|
||||
|
||||
bool GraphData::IsEmpty() const
|
||||
{
|
||||
// Note. |m_transfers| may be empty if GraphData instance is not empty.
|
||||
return m_stops.empty() || m_gates.empty() || m_edges.empty() || m_lines.empty()
|
||||
|| m_shapes.empty() || m_networks.empty();
|
||||
}
|
||||
|
||||
void GraphData::Sort()
|
||||
{
|
||||
SortVisitor const v{};
|
||||
Visit(v);
|
||||
}
|
||||
|
||||
void GraphData::ClipGraph(vector<m2::RegionD> const & borders)
|
||||
{
|
||||
Sort();
|
||||
CHECK(IsValid(), ());
|
||||
ClipLines(borders);
|
||||
ClipStops();
|
||||
ClipNetworks();
|
||||
ClipGates();
|
||||
ClipTransfer();
|
||||
ClipEdges();
|
||||
ClipShapes();
|
||||
CHECK(IsValid(), ());
|
||||
}
|
||||
|
||||
void GraphData::CalculateBestPedestrianSegments(string const & mwmPath, TCountryId const & countryId)
|
||||
/// \brief Calculates best pedestrian segment for every gate in |m_gates|.
|
||||
/// \note All gates in |m_gates| must have a valid |m_point| field before the call.
|
||||
void CalculateBestPedestrianSegments(string const & mwmPath, TCountryId const & countryId,
|
||||
GraphData & graphData)
|
||||
{
|
||||
// Creating IndexRouter.
|
||||
SingleMwmIndex index(mwmPath);
|
||||
|
@ -446,8 +97,10 @@ void GraphData::CalculateBestPedestrianSegments(string const & mwmPath, TCountry
|
|||
auto worldGraph = indexRouter.MakeSingleMwmWorldGraph();
|
||||
|
||||
// Looking for the best segment for every gate.
|
||||
for (auto & gate : m_gates)
|
||||
auto const & gates = graphData.GetGates();
|
||||
for (size_t i = 0; i < gates.size(); ++i)
|
||||
{
|
||||
auto const & gate = gates[i];
|
||||
if (countryFileGetter(gate.GetPoint()) != countryId)
|
||||
continue;
|
||||
// Note. For pedestrian routing all the segments are considered as twoway segments so
|
||||
|
@ -462,7 +115,7 @@ void GraphData::CalculateBestPedestrianSegments(string const & mwmPath, TCountry
|
|||
true /* isOutgoing */, *worldGraph, bestSegment))
|
||||
{
|
||||
CHECK_EQUAL(bestSegment.GetMwmId(), 0, ());
|
||||
gate.SetBestPedestrianSegment(SingleMwmSegment(
|
||||
graphData.SetGateBestPedestrianSegment(i, SingleMwmSegment(
|
||||
bestSegment.GetFeatureId(), bestSegment.GetSegmentIdx(), bestSegment.IsForward()));
|
||||
}
|
||||
}
|
||||
|
@ -478,150 +131,12 @@ void GraphData::CalculateBestPedestrianSegments(string const & mwmPath, TCountry
|
|||
}
|
||||
}
|
||||
}
|
||||
} // namespace
|
||||
|
||||
bool GraphData::IsUnique() const
|
||||
namespace routing
|
||||
{
|
||||
IsUniqueVisitor v;
|
||||
Visit(v);
|
||||
return v.IsUnique();
|
||||
}
|
||||
|
||||
bool GraphData::IsSorted() const
|
||||
namespace transit
|
||||
{
|
||||
IsSortedVisitor v;
|
||||
Visit(v);
|
||||
return v.IsSorted();
|
||||
}
|
||||
|
||||
void GraphData::ClipLines(vector<m2::RegionD> const & borders)
|
||||
{
|
||||
// Set with stop ids with stops which are inside |borders|.
|
||||
set<StopId> stopIdInside;
|
||||
for (auto const & stop : m_stops)
|
||||
{
|
||||
if (m2::RegionsContain(borders, stop.GetPoint()))
|
||||
stopIdInside.insert(stop.GetId());
|
||||
}
|
||||
|
||||
set<StopId> hasNeighborInside;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
auto const stop1Inside = stopIdInside.count(edge.GetStop1Id()) != 0;
|
||||
auto const stop2Inside = stopIdInside.count(edge.GetStop2Id()) != 0;
|
||||
if (stop1Inside && !stop2Inside)
|
||||
hasNeighborInside.insert(edge.GetStop2Id());
|
||||
if (stop2Inside && !stop1Inside)
|
||||
hasNeighborInside.insert(edge.GetStop1Id());
|
||||
}
|
||||
|
||||
stopIdInside.insert(hasNeighborInside.cbegin(), hasNeighborInside.cend());
|
||||
|
||||
// Filling |lines| with stops inside |borders|.
|
||||
vector<Line> lines;
|
||||
for (auto const & line : m_lines)
|
||||
{
|
||||
// Note. |stopIdsToFill| will be filled with continuous sequences of stop ids.
|
||||
// In most cases only one sequence of stop ids should be placed to |stopIdsToFill|.
|
||||
// But if a line is split by |borders| several times then several
|
||||
// continuous groups of stop ids will be placed to |stopIdsToFill|.
|
||||
// The loop below goes through all the stop ids belong the line |line| and
|
||||
// keeps in |stopIdsToFill| continuous groups of stop ids which are inside |borders|.
|
||||
Ranges stopIdsToFill;
|
||||
Ranges const & ranges = line.GetStopIds();
|
||||
CHECK_EQUAL(ranges.size(), 1, ());
|
||||
vector<StopId> const & stopIds = ranges[0];
|
||||
auto it = stopIds.begin();
|
||||
while (it != stopIds.end()) {
|
||||
while (it != stopIds.end() && stopIdInside.count(*it) == 0)
|
||||
++it;
|
||||
auto jt = it;
|
||||
while (jt != stopIds.end() && stopIdInside.count(*jt) != 0)
|
||||
++jt;
|
||||
if (it != jt)
|
||||
stopIdsToFill.emplace_back(it, jt);
|
||||
it = jt;
|
||||
}
|
||||
|
||||
if (!stopIdsToFill.empty())
|
||||
{
|
||||
lines.emplace_back(line.GetId(), line.GetNumber(), line.GetTitle(), line.GetType(),
|
||||
line.GetColor(), line.GetNetworkId(), stopIdsToFill, line.GetInterval());
|
||||
}
|
||||
}
|
||||
|
||||
m_lines.swap(lines);
|
||||
}
|
||||
|
||||
void GraphData::ClipStops()
|
||||
{
|
||||
CHECK(is_sorted(m_stops.cbegin(), m_stops.cend()), ());
|
||||
set<StopId> stopIds;
|
||||
for (auto const & line : m_lines)
|
||||
{
|
||||
for (auto const & range : line.GetStopIds())
|
||||
stopIds.insert(range.cbegin(), range.cend());
|
||||
}
|
||||
|
||||
UpdateItems(stopIds, m_stops);
|
||||
}
|
||||
|
||||
void GraphData::ClipNetworks()
|
||||
{
|
||||
CHECK(is_sorted(m_networks.cbegin(), m_networks.cend()), ());
|
||||
set<NetworkId> networkIds;
|
||||
for (auto const & line : m_lines)
|
||||
networkIds.insert(line.GetNetworkId());
|
||||
|
||||
UpdateItems(networkIds, m_networks);
|
||||
}
|
||||
|
||||
void GraphData::ClipGates()
|
||||
{
|
||||
ClipItemsByStops(m_stops, m_gates);
|
||||
}
|
||||
|
||||
void GraphData::ClipTransfer()
|
||||
{
|
||||
ClipItemsByStops(m_stops, m_transfers);
|
||||
}
|
||||
|
||||
void GraphData::ClipEdges()
|
||||
{
|
||||
CHECK(is_sorted(m_stops.cbegin(), m_stops.cend()), ());
|
||||
|
||||
vector<Edge> edges;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
if (HasStop(m_stops, edge.GetStop1Id()) && HasStop(m_stops, edge.GetStop2Id()))
|
||||
edges.push_back(edge);
|
||||
}
|
||||
|
||||
SortVisitor{}(edges, nullptr /* name */);
|
||||
m_edges.swap(edges);
|
||||
}
|
||||
|
||||
void GraphData::ClipShapes()
|
||||
{
|
||||
CHECK(is_sorted(m_edges.cbegin(), m_edges.cend()), ());
|
||||
|
||||
// Set with shape ids contained in m_edges.
|
||||
set<ShapeId> shapeIdInEdges;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
auto const & shapeIds = edge.GetShapeIds();
|
||||
shapeIdInEdges.insert(shapeIds.cbegin(), shapeIds.cend());
|
||||
}
|
||||
|
||||
vector<Shape> shapes;
|
||||
for (auto const & shape : m_shapes)
|
||||
{
|
||||
if (shapeIdInEdges.count(shape.GetId()) != 0)
|
||||
shapes.push_back(shape);
|
||||
}
|
||||
|
||||
m_shapes.swap(shapes);
|
||||
}
|
||||
|
||||
void DeserializeFromJson(OsmIdToFeatureIdsMap const & mapping,
|
||||
string const & transitJsonPath, GraphData & data)
|
||||
{
|
||||
|
@ -658,7 +173,7 @@ void DeserializeFromJson(OsmIdToFeatureIdsMap const & mapping,
|
|||
void ProcessGraph(string const & mwmPath, TCountryId const & countryId,
|
||||
OsmIdToFeatureIdsMap const & osmIdToFeatureIdsMap, GraphData & data)
|
||||
{
|
||||
data.CalculateBestPedestrianSegments(mwmPath, countryId);
|
||||
CalculateBestPedestrianSegments(mwmPath, countryId, data);
|
||||
data.Sort();
|
||||
CHECK(data.IsValid(), (mwmPath));
|
||||
}
|
||||
|
|
|
@ -1,200 +1,15 @@
|
|||
#pragma once
|
||||
|
||||
#include "routing_common/transit_types.hpp"
|
||||
#include "routing_common/transit_graph_data.hpp"
|
||||
|
||||
#include "storage/index.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
#include "geometry/region2d.hpp"
|
||||
|
||||
#include "coding/reader.hpp"
|
||||
#include "coding/writer.hpp"
|
||||
|
||||
#include "base/macros.hpp"
|
||||
#include "base/osm_id.hpp"
|
||||
|
||||
#include "3party/jansson/myjansson.hpp"
|
||||
|
||||
#include <cstdint>
|
||||
#include <cstring>
|
||||
#include <map>
|
||||
#include <string>
|
||||
#include <type_traits>
|
||||
#include <vector>
|
||||
|
||||
namespace routing
|
||||
{
|
||||
namespace transit
|
||||
{
|
||||
using OsmIdToFeatureIdsMap = std::map<osm::Id, std::vector<FeatureId>>;
|
||||
|
||||
class DeserializerFromJson
|
||||
{
|
||||
public:
|
||||
DeserializerFromJson(json_struct_t * node, OsmIdToFeatureIdsMap const & osmIdToFeatureIds);
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_integral<T>::value || std::is_enum<T>::value ||
|
||||
std::is_same<T, double>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
GetField(t, name);
|
||||
return;
|
||||
}
|
||||
|
||||
void operator()(std::string & s, char const * name = nullptr) { GetField(s, name); }
|
||||
void operator()(m2::PointD & p, char const * name = nullptr);
|
||||
void operator()(FeatureIdentifiers & id, char const * name = nullptr);
|
||||
void operator()(EdgeFlags & edgeFlags, char const * name = nullptr);
|
||||
void operator()(StopIdRanges & rs, char const * name = nullptr);
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_same<T, Edge::WrappedEdgeId>::value ||
|
||||
std::is_same<T, Stop::WrappedStopId>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
typename T::RepType id;
|
||||
operator()(id, name);
|
||||
t.Set(id);
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void operator()(std::vector<T> & vs, char const * name = nullptr)
|
||||
{
|
||||
auto * arr = my::GetJSONOptionalField(m_node, name);
|
||||
if (arr == nullptr)
|
||||
return;
|
||||
|
||||
if (!json_is_array(arr))
|
||||
MYTHROW(my::Json::Exception, ("The field", name, "must contain a json array."));
|
||||
size_t const sz = json_array_size(arr);
|
||||
vs.resize(sz);
|
||||
for (size_t i = 0; i < sz; ++i)
|
||||
{
|
||||
DeserializerFromJson arrayItem(json_array_get(arr, i), m_osmIdToFeatureIds);
|
||||
arrayItem(vs[i]);
|
||||
}
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_class<T>::value &&
|
||||
!std::is_same<T, Edge::WrappedEdgeId>::value &&
|
||||
!std::is_same<T, Stop::WrappedStopId>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
if (name != nullptr && json_is_object(m_node))
|
||||
{
|
||||
json_t * dictNode = my::GetJSONOptionalField(m_node, name);
|
||||
if (dictNode == nullptr)
|
||||
return; // No such field in json.
|
||||
|
||||
DeserializerFromJson dict(dictNode, m_osmIdToFeatureIds);
|
||||
t.Visit(dict);
|
||||
return;
|
||||
}
|
||||
|
||||
t.Visit(*this);
|
||||
}
|
||||
|
||||
private:
|
||||
template <typename T>
|
||||
void GetField(T & t, char const * name = nullptr)
|
||||
{
|
||||
if (name == nullptr)
|
||||
{
|
||||
// |name| is not set in case of array items
|
||||
FromJSON(m_node, t);
|
||||
return;
|
||||
}
|
||||
|
||||
json_struct_t * field = my::GetJSONOptionalField(m_node, name);
|
||||
if (field == nullptr)
|
||||
{
|
||||
// No optional field |name| at |m_node|. In that case the default value should be set to |t|.
|
||||
// This default value is set at constructor of corresponding class which is filled with
|
||||
// |DeserializerFromJson|. And the value (|t|) is not changed at this method.
|
||||
return;
|
||||
}
|
||||
FromJSON(field, t);
|
||||
}
|
||||
|
||||
json_struct_t * m_node;
|
||||
OsmIdToFeatureIdsMap const & m_osmIdToFeatureIds;
|
||||
};
|
||||
|
||||
/// \brief The class contains all the information to make TRANSIT_FILE_TAG section.
|
||||
class GraphData
|
||||
{
|
||||
public:
|
||||
void DeserializeFromJson(my::Json const & root, OsmIdToFeatureIdsMap const & mapping);
|
||||
void Serialize(Writer & writer) const;
|
||||
void Deserialize(Reader & reader);
|
||||
void AppendTo(GraphData const & rhs);
|
||||
void Clear();
|
||||
bool IsValid() const;
|
||||
bool IsEmpty() const;
|
||||
|
||||
/// \brief Sorts all class fields by their ids.
|
||||
void Sort();
|
||||
/// \brief Removes some items from all the class fields if they are outside |borders|.
|
||||
/// Please see description for the other Clip*() method for excact rules of clipping.
|
||||
/// \note Before call of the method every line in |m_stopIds| should contain |m_stopIds|
|
||||
/// with only one stop range.
|
||||
void ClipGraph(std::vector<m2::RegionD> const & borders);
|
||||
/// \brief Calculates best pedestrian segment for every gate in |m_gates|.
|
||||
/// \note All gates in |m_gates| must have a valid |m_point| field before the call.
|
||||
void CalculateBestPedestrianSegments(std::string const & mwmPath, std::string const & countryId);
|
||||
|
||||
std::vector<Stop> const & GetStops() const { return m_stops; }
|
||||
std::vector<Gate> const & GetGates() const { return m_gates; }
|
||||
std::vector<Edge> const & GetEdges() const { return m_edges; }
|
||||
std::vector<Transfer> const & GetTransfers() const { return m_transfers; }
|
||||
std::vector<Line> const & GetLines() const { return m_lines; }
|
||||
std::vector<Shape> const & GetShapes() const { return m_shapes; }
|
||||
std::vector<Network> const & GetNetworks() const { return m_networks; }
|
||||
|
||||
private:
|
||||
DECLARE_VISITOR_AND_DEBUG_PRINT(GraphData, visitor(m_stops, "stops"), visitor(m_gates, "gates"),
|
||||
visitor(m_edges, "edges"), visitor(m_transfers, "transfers"),
|
||||
visitor(m_lines, "lines"), visitor(m_shapes, "shapes"),
|
||||
visitor(m_networks, "networks"))
|
||||
|
||||
bool IsUnique() const;
|
||||
bool IsSorted() const;
|
||||
|
||||
/// \brief Clipping |m_lines| with |borders|.
|
||||
/// \details After a call of the method the following stop ids in |m_lines| are left:
|
||||
/// * stops inside |borders|
|
||||
/// * stops which are connected with an edge from |m_edges| with stops inside |borders|
|
||||
/// \note Lines without stops are removed from |m_lines|.
|
||||
/// \note Before call of the method every line in |m_lines| should contain |m_stopIds|
|
||||
/// with only one stop range.
|
||||
void ClipLines(std::vector<m2::RegionD> const & borders);
|
||||
/// \brief Removes all stops from |m_stops| which are not contained in |m_lines| at field |m_stopIds|.
|
||||
/// \note Only stops which stop ids contained in |m_lines| will left in |m_stops|
|
||||
/// after call of this method.
|
||||
void ClipStops();
|
||||
/// \brief Removes all networks from |m_networks| which are not contained in |m_lines|
|
||||
/// at field |m_networkId|.
|
||||
void ClipNetworks();
|
||||
/// \brief Removes gates from |m_gates| if there's no stop in |m_stops| with their stop ids.
|
||||
void ClipGates();
|
||||
/// \brief Removes transfers from |m_transfers| if there's no stop in |m_stops| with their stop ids.
|
||||
void ClipTransfer();
|
||||
/// \brief Removes edges from |m_edges| if their ends are not contained in |m_stops|.
|
||||
void ClipEdges();
|
||||
/// \brief Removes all shapes from |m_shapes| which are not reffered form |m_edges|.
|
||||
void ClipShapes();
|
||||
|
||||
std::vector<Stop> m_stops;
|
||||
std::vector<Gate> m_gates;
|
||||
std::vector<Edge> m_edges;
|
||||
std::vector<Transfer> m_transfers;
|
||||
std::vector<Line> m_lines;
|
||||
std::vector<Shape> m_shapes;
|
||||
std::vector<Network> m_networks;
|
||||
};
|
||||
|
||||
/// \brief Fills |data| according to a transit graph (|transitJsonPath|).
|
||||
/// \note Some fields of |data| contain feature ids of a certain mwm. These fields are filled
|
||||
/// iff the mapping (|osmIdToFeatureIdsPath|) contains them. Otherwise the fields have default value.
|
||||
|
@ -207,7 +22,7 @@ void ProcessGraph(std::string const & mwmPath, storage::TCountryId const & count
|
|||
OsmIdToFeatureIdsMap const & osmIdToFeatureIdsMap, GraphData & data);
|
||||
|
||||
/// \brief Builds the transit section in the mwm based on transit graph in json which represents
|
||||
/// trasit graph clipped by the mwm borders.
|
||||
/// transit graph clipped by the mwm borders.
|
||||
/// \param mwmDir relative or full path to a directory where mwm is located.
|
||||
/// \param countryId is an mwm name without extension of the processed mwm.
|
||||
/// \param osmIdToFeatureIdsPath is a path to a file with osm id to feature ids mapping.
|
||||
|
|
|
@ -1,5 +1,9 @@
|
|||
project(routing_common)
|
||||
|
||||
include_directories(
|
||||
${OMIM_ROOT}/3party/jansson/src
|
||||
)
|
||||
|
||||
set(
|
||||
SRC
|
||||
bicycle_model.cpp
|
||||
|
@ -9,6 +13,8 @@ set(
|
|||
num_mwm_id.hpp
|
||||
pedestrian_model.cpp
|
||||
pedestrian_model.hpp
|
||||
transit_graph_data.cpp
|
||||
transit_graph_data.hpp
|
||||
transit_serdes.hpp
|
||||
transit_speed_limits.hpp
|
||||
transit_types.cpp
|
||||
|
|
|
@ -16,6 +16,7 @@ SOURCES += \
|
|||
bicycle_model.cpp \
|
||||
car_model.cpp \
|
||||
pedestrian_model.cpp \
|
||||
transit_graph_data.cpp \
|
||||
transit_types.cpp \
|
||||
vehicle_model.cpp \
|
||||
|
||||
|
@ -25,6 +26,7 @@ HEADERS += \
|
|||
car_model.hpp \
|
||||
num_mwm_id.hpp \
|
||||
pedestrian_model.hpp \
|
||||
transit_graph_data.hpp \
|
||||
transit_serdes.hpp \
|
||||
transit_speed_limits.hpp \
|
||||
transit_types.hpp \
|
||||
|
|
509
routing_common/transit_graph_data.cpp
Normal file
509
routing_common/transit_graph_data.cpp
Normal file
|
@ -0,0 +1,509 @@
|
|||
#include "routing_common/transit_graph_data.hpp"
|
||||
|
||||
#include "routing_common/transit_serdes.hpp"
|
||||
|
||||
#include "base/assert.hpp"
|
||||
#include "base/checked_cast.hpp"
|
||||
#include "base/logging.hpp"
|
||||
#include "base/stl_helpers.hpp"
|
||||
#include "base/string_utils.hpp"
|
||||
|
||||
#include <algorithm>
|
||||
#include <iterator>
|
||||
#include <set>
|
||||
|
||||
#include "defines.hpp"
|
||||
|
||||
using namespace routing;
|
||||
using namespace routing::transit;
|
||||
using namespace std;
|
||||
|
||||
namespace
|
||||
{
|
||||
struct ClearVisitor
|
||||
{
|
||||
template<typename Cont>
|
||||
void operator()(Cont & c, char const * /* name */) const { c.clear(); }
|
||||
};
|
||||
|
||||
struct SortVisitor
|
||||
{
|
||||
template<typename Cont>
|
||||
void operator()(Cont & c, char const * /* name */) const
|
||||
{
|
||||
sort(c.begin(), c.end());
|
||||
}
|
||||
};
|
||||
|
||||
struct IsValidVisitor
|
||||
{
|
||||
template<typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */ )
|
||||
{
|
||||
m_isValid = m_isValid && ::IsValid(c);
|
||||
}
|
||||
|
||||
bool IsValid() const { return m_isValid; }
|
||||
|
||||
private:
|
||||
bool m_isValid = true;
|
||||
};
|
||||
|
||||
struct IsUniqueVisitor
|
||||
{
|
||||
template<typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */)
|
||||
{
|
||||
m_isUnique = m_isUnique && (adjacent_find(c.cbegin(), c.cend()) == c.cend());
|
||||
}
|
||||
|
||||
bool IsUnique() const { return m_isUnique; }
|
||||
|
||||
private:
|
||||
bool m_isUnique = true;
|
||||
};
|
||||
|
||||
struct IsSortedVisitor
|
||||
{
|
||||
template<typename Cont>
|
||||
void operator()(Cont const & c, char const * /* name */)
|
||||
{
|
||||
m_isSorted = m_isSorted && is_sorted(c.cbegin(), c.cend());
|
||||
}
|
||||
|
||||
bool IsSorted() const { return m_isSorted; }
|
||||
|
||||
private:
|
||||
bool m_isSorted = true;
|
||||
};
|
||||
|
||||
template<typename T>
|
||||
void Append(vector<T> const & src, vector<T> & dst)
|
||||
{
|
||||
dst.insert(dst.end(), src.begin(), src.end());
|
||||
}
|
||||
|
||||
bool HasStop(vector<Stop> const & stops, StopId stopId)
|
||||
{
|
||||
return binary_search(stops.cbegin(), stops.cend(), Stop(stopId));
|
||||
}
|
||||
|
||||
/// \brief Removes from |items| all items which stop ids is not contained in |stops|.
|
||||
/// \note This method keeps relative order of |items|.
|
||||
template <class Item>
|
||||
void ClipItemsByStops(vector<Stop> const & stops, vector<Item> & items)
|
||||
{
|
||||
CHECK(is_sorted(stops.cbegin(), stops.cend()), ());
|
||||
|
||||
vector<Item> itemsToFill;
|
||||
for (auto const & item : items)
|
||||
{
|
||||
for (auto const stopId : item.GetStopIds())
|
||||
{
|
||||
if (HasStop(stops, stopId))
|
||||
{
|
||||
itemsToFill.push_back(item);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
items.swap(itemsToFill);
|
||||
}
|
||||
|
||||
/// \returns ref to an item at |items| by |id|.
|
||||
/// \note |items| must be sorted before a call of this method.
|
||||
template <class Id, class Item>
|
||||
Item const & FindById(vector<Item> const & items, Id id)
|
||||
{
|
||||
auto const s1Id = equal_range(items.cbegin(), items.cend(), Item(id));
|
||||
CHECK_EQUAL(distance(s1Id.first, s1Id.second), 1,
|
||||
("An item with id:", id, "is not unique or there's not such item. items:", items));
|
||||
return *s1Id.first;
|
||||
}
|
||||
|
||||
/// \brief Fills |items| with items which have ids from |ids|.
|
||||
/// \note |items| must be sorted before a call of this method.
|
||||
template <class Id, class Item>
|
||||
void UpdateItems(set<Id> const & ids, vector<Item> & items)
|
||||
{
|
||||
vector<Item> itemsToFill;
|
||||
for (auto const id : ids)
|
||||
itemsToFill.push_back(FindById(items, id));
|
||||
|
||||
SortVisitor{}(itemsToFill, nullptr /* name */);
|
||||
items.swap(itemsToFill);
|
||||
}
|
||||
} // namespace
|
||||
|
||||
namespace routing
|
||||
{
|
||||
namespace transit
|
||||
{
|
||||
// DeserializerFromJson ---------------------------------------------------------------------------
|
||||
DeserializerFromJson::DeserializerFromJson(json_struct_t * node,
|
||||
OsmIdToFeatureIdsMap const & osmIdToFeatureIds)
|
||||
: m_node(node), m_osmIdToFeatureIds(osmIdToFeatureIds)
|
||||
{
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(m2::PointD & p, char const * name)
|
||||
{
|
||||
// @todo(bykoianko) Instead of having a special operator() method for m2::PointD class it's
|
||||
// necessary to add Point class to transit_types.hpp and process it in DeserializerFromJson with
|
||||
// regular method.
|
||||
json_t * item = nullptr;
|
||||
if (name == nullptr)
|
||||
item = m_node; // Array item case
|
||||
else
|
||||
item = my::GetJSONObligatoryField(m_node, name);
|
||||
|
||||
CHECK(json_is_object(item), ("Item is not a json object:", name));
|
||||
FromJSONObject(item, "x", p.x);
|
||||
FromJSONObject(item, "y", p.y);
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(FeatureIdentifiers & id, char const * name)
|
||||
{
|
||||
// Conversion osm id to feature id.
|
||||
string osmIdStr;
|
||||
GetField(osmIdStr, name);
|
||||
CHECK(strings::is_number(osmIdStr), ("Osm id string is not a number:", osmIdStr));
|
||||
uint64_t osmIdNum;
|
||||
CHECK(strings::to_uint64(osmIdStr, osmIdNum),
|
||||
("Cann't convert osm id string:", osmIdStr, "to a number."));
|
||||
osm::Id const osmId(osmIdNum);
|
||||
auto const it = m_osmIdToFeatureIds.find(osmId);
|
||||
if (it != m_osmIdToFeatureIds.cend())
|
||||
{
|
||||
CHECK_EQUAL(it->second.size(), 1, ("Osm id:", osmId, "(encoded", osmId.EncodedId(),
|
||||
") from transit graph corresponds to", it->second.size(), "features."
|
||||
"But osm id should be represented be one feature."));
|
||||
id.SetFeatureId(it->second[0]);
|
||||
}
|
||||
id.SetOsmId(osmId.EncodedId());
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(EdgeFlags & edgeFlags, char const * name)
|
||||
{
|
||||
bool transfer = false;
|
||||
(*this)(transfer, name);
|
||||
// Note. Only |transfer| field of |edgeFlags| may be set at this point because the
|
||||
// other fields of |edgeFlags| are unknown.
|
||||
edgeFlags.SetFlags(0);
|
||||
edgeFlags.m_transfer = transfer;
|
||||
}
|
||||
|
||||
void DeserializerFromJson::operator()(StopIdRanges & rs, char const * name)
|
||||
{
|
||||
vector<StopId> stopIds;
|
||||
(*this)(stopIds, name);
|
||||
rs = StopIdRanges({stopIds});
|
||||
}
|
||||
|
||||
// GraphData --------------------------------------------------------------------------------------
|
||||
void GraphData::DeserializeFromJson(my::Json const & root, OsmIdToFeatureIdsMap const & mapping)
|
||||
{
|
||||
DeserializerFromJson deserializer(root.get(), mapping);
|
||||
Visit(deserializer);
|
||||
|
||||
// Removes equivalent edges from |m_edges|. If there are several equivalent edges only
|
||||
// the most lightweight edge is left.
|
||||
// Note. It's possible that two stops are connected with the same line several times
|
||||
// in the same direction. It happens in Oslo metro (T-banen):
|
||||
// https://en.wikipedia.org/wiki/Oslo_Metro#/media/File:Oslo_Metro_Map.svg branch 5.
|
||||
my::SortUnique(m_edges,
|
||||
[](Edge const & e1, Edge const & e2) {
|
||||
if (e1 != e2)
|
||||
return e1 < e2;
|
||||
return e1.GetWeight() < e2.GetWeight();
|
||||
},
|
||||
[](Edge const & e1, Edge const & e2) { return e1 == e2; });
|
||||
}
|
||||
|
||||
void GraphData::Serialize(Writer & writer) const
|
||||
{
|
||||
TransitHeader header;
|
||||
|
||||
auto const startOffset = writer.Pos();
|
||||
Serializer<Writer> serializer(writer);
|
||||
FixedSizeSerializer<Writer> numberSerializer(writer);
|
||||
numberSerializer(header);
|
||||
header.m_stopsOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_stops);
|
||||
header.m_gatesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_gates);
|
||||
header.m_edgesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_edges);
|
||||
header.m_transfersOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_transfers);
|
||||
header.m_linesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_lines);
|
||||
header.m_shapesOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_shapes);
|
||||
header.m_networksOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
serializer(m_networks);
|
||||
header.m_endOffset = base::checked_cast<uint32_t>(writer.Pos() - startOffset);
|
||||
|
||||
// Rewriting header info.
|
||||
CHECK(header.IsValid(), (header));
|
||||
auto const endOffset = writer.Pos();
|
||||
writer.Seek(startOffset);
|
||||
numberSerializer(header);
|
||||
writer.Seek(endOffset);
|
||||
|
||||
LOG(LINFO, (TRANSIT_FILE_TAG, "section is ready. Header:", header));
|
||||
}
|
||||
|
||||
void GraphData::Deserialize(Reader & reader)
|
||||
{
|
||||
NonOwningReaderSource src(reader);
|
||||
transit::Deserializer<NonOwningReaderSource> deserializer(src);
|
||||
transit::FixedSizeDeserializer<NonOwningReaderSource> numberDeserializer(src);
|
||||
|
||||
transit::TransitHeader header;
|
||||
numberDeserializer(header);
|
||||
CHECK(header.IsValid(), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_stopsOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_stops);
|
||||
CHECK(transit::IsValidSortedUnique(m_stops), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_gatesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_gates);
|
||||
CHECK(transit::IsValidSortedUnique(m_gates), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_edgesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_edges);
|
||||
CHECK(transit::IsValidSortedUnique(m_edges), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_transfersOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_transfers);
|
||||
CHECK(transit::IsValidSortedUnique(m_transfers), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_linesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_lines);
|
||||
CHECK(transit::IsValidSortedUnique(m_lines), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_shapesOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_shapes);
|
||||
CHECK(transit::IsValidSortedUnique(m_shapes), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_networksOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
deserializer(m_networks);
|
||||
CHECK(transit::IsValidSortedUnique(m_networks), ());
|
||||
|
||||
CHECK_EQUAL(src.Pos(), header.m_endOffset, ("Wrong", TRANSIT_FILE_TAG, "section format."));
|
||||
}
|
||||
|
||||
void GraphData::AppendTo(GraphData const & rhs)
|
||||
{
|
||||
::Append(rhs.m_stops, m_stops);
|
||||
::Append(rhs.m_gates, m_gates);
|
||||
::Append(rhs.m_edges, m_edges);
|
||||
::Append(rhs.m_transfers, m_transfers);
|
||||
::Append(rhs.m_lines, m_lines);
|
||||
::Append(rhs.m_shapes, m_shapes);
|
||||
::Append(rhs.m_networks, m_networks);
|
||||
}
|
||||
|
||||
void GraphData::Clear()
|
||||
{
|
||||
ClearVisitor const v{};
|
||||
Visit(v);
|
||||
}
|
||||
|
||||
bool GraphData::IsValid() const
|
||||
{
|
||||
if (!IsSorted() || !IsUnique())
|
||||
return false;
|
||||
|
||||
IsValidVisitor v;
|
||||
Visit(v);
|
||||
return v.IsValid();
|
||||
}
|
||||
|
||||
bool GraphData::IsEmpty() const
|
||||
{
|
||||
// Note. |m_transfers| may be empty if GraphData instance is not empty.
|
||||
return m_stops.empty() || m_gates.empty() || m_edges.empty() || m_lines.empty()
|
||||
|| m_shapes.empty() || m_networks.empty();
|
||||
}
|
||||
|
||||
void GraphData::Sort()
|
||||
{
|
||||
SortVisitor const v{};
|
||||
Visit(v);
|
||||
}
|
||||
|
||||
void GraphData::ClipGraph(vector<m2::RegionD> const & borders)
|
||||
{
|
||||
Sort();
|
||||
CHECK(IsValid(), ());
|
||||
ClipLines(borders);
|
||||
ClipStops();
|
||||
ClipNetworks();
|
||||
ClipGates();
|
||||
ClipTransfer();
|
||||
ClipEdges();
|
||||
ClipShapes();
|
||||
CHECK(IsValid(), ());
|
||||
}
|
||||
|
||||
void GraphData::SetGateBestPedestrianSegment(size_t gateIdx, SingleMwmSegment const & s)
|
||||
{
|
||||
CHECK_LESS(gateIdx, m_gates.size(), ());
|
||||
m_gates[gateIdx].SetBestPedestrianSegment(s);
|
||||
}
|
||||
|
||||
bool GraphData::IsUnique() const
|
||||
{
|
||||
IsUniqueVisitor v;
|
||||
Visit(v);
|
||||
return v.IsUnique();
|
||||
}
|
||||
|
||||
bool GraphData::IsSorted() const
|
||||
{
|
||||
IsSortedVisitor v;
|
||||
Visit(v);
|
||||
return v.IsSorted();
|
||||
}
|
||||
|
||||
void GraphData::ClipLines(vector<m2::RegionD> const & borders)
|
||||
{
|
||||
// Set with stop ids with stops which are inside |borders|.
|
||||
set<StopId> stopIdInside;
|
||||
for (auto const & stop : m_stops)
|
||||
{
|
||||
if (m2::RegionsContain(borders, stop.GetPoint()))
|
||||
stopIdInside.insert(stop.GetId());
|
||||
}
|
||||
|
||||
set<StopId> hasNeighborInside;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
auto const stop1Inside = stopIdInside.count(edge.GetStop1Id()) != 0;
|
||||
auto const stop2Inside = stopIdInside.count(edge.GetStop2Id()) != 0;
|
||||
if (stop1Inside && !stop2Inside)
|
||||
hasNeighborInside.insert(edge.GetStop2Id());
|
||||
if (stop2Inside && !stop1Inside)
|
||||
hasNeighborInside.insert(edge.GetStop1Id());
|
||||
}
|
||||
|
||||
stopIdInside.insert(hasNeighborInside.cbegin(), hasNeighborInside.cend());
|
||||
|
||||
// Filling |lines| with stops inside |borders|.
|
||||
vector<Line> lines;
|
||||
for (auto const & line : m_lines)
|
||||
{
|
||||
// Note. |stopIdsToFill| will be filled with continuous sequences of stop ids.
|
||||
// In most cases only one sequence of stop ids should be placed to |stopIdsToFill|.
|
||||
// But if a line is split by |borders| several times then several
|
||||
// continuous groups of stop ids will be placed to |stopIdsToFill|.
|
||||
// The loop below goes through all the stop ids belong the line |line| and
|
||||
// keeps in |stopIdsToFill| continuous groups of stop ids which are inside |borders|.
|
||||
Ranges stopIdsToFill;
|
||||
Ranges const & ranges = line.GetStopIds();
|
||||
CHECK_EQUAL(ranges.size(), 1, ());
|
||||
vector<StopId> const & stopIds = ranges[0];
|
||||
auto it = stopIds.begin();
|
||||
while (it != stopIds.end()) {
|
||||
while (it != stopIds.end() && stopIdInside.count(*it) == 0)
|
||||
++it;
|
||||
auto jt = it;
|
||||
while (jt != stopIds.end() && stopIdInside.count(*jt) != 0)
|
||||
++jt;
|
||||
if (it != jt)
|
||||
stopIdsToFill.emplace_back(it, jt);
|
||||
it = jt;
|
||||
}
|
||||
|
||||
if (!stopIdsToFill.empty())
|
||||
{
|
||||
lines.emplace_back(line.GetId(), line.GetNumber(), line.GetTitle(), line.GetType(),
|
||||
line.GetColor(), line.GetNetworkId(), stopIdsToFill, line.GetInterval());
|
||||
}
|
||||
}
|
||||
|
||||
m_lines.swap(lines);
|
||||
}
|
||||
|
||||
void GraphData::ClipStops()
|
||||
{
|
||||
CHECK(is_sorted(m_stops.cbegin(), m_stops.cend()), ());
|
||||
set<StopId> stopIds;
|
||||
for (auto const & line : m_lines)
|
||||
{
|
||||
for (auto const & range : line.GetStopIds())
|
||||
stopIds.insert(range.cbegin(), range.cend());
|
||||
}
|
||||
|
||||
UpdateItems(stopIds, m_stops);
|
||||
}
|
||||
|
||||
void GraphData::ClipNetworks()
|
||||
{
|
||||
CHECK(is_sorted(m_networks.cbegin(), m_networks.cend()), ());
|
||||
set<NetworkId> networkIds;
|
||||
for (auto const & line : m_lines)
|
||||
networkIds.insert(line.GetNetworkId());
|
||||
|
||||
UpdateItems(networkIds, m_networks);
|
||||
}
|
||||
|
||||
void GraphData::ClipGates()
|
||||
{
|
||||
ClipItemsByStops(m_stops, m_gates);
|
||||
}
|
||||
|
||||
void GraphData::ClipTransfer()
|
||||
{
|
||||
ClipItemsByStops(m_stops, m_transfers);
|
||||
}
|
||||
|
||||
void GraphData::ClipEdges()
|
||||
{
|
||||
CHECK(is_sorted(m_stops.cbegin(), m_stops.cend()), ());
|
||||
|
||||
vector<Edge> edges;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
if (HasStop(m_stops, edge.GetStop1Id()) && HasStop(m_stops, edge.GetStop2Id()))
|
||||
edges.push_back(edge);
|
||||
}
|
||||
|
||||
SortVisitor{}(edges, nullptr /* name */);
|
||||
m_edges.swap(edges);
|
||||
}
|
||||
|
||||
void GraphData::ClipShapes()
|
||||
{
|
||||
CHECK(is_sorted(m_edges.cbegin(), m_edges.cend()), ());
|
||||
|
||||
// Set with shape ids contained in m_edges.
|
||||
set<ShapeId> shapeIdInEdges;
|
||||
for (auto const & edge : m_edges)
|
||||
{
|
||||
auto const & shapeIds = edge.GetShapeIds();
|
||||
shapeIdInEdges.insert(shapeIds.cbegin(), shapeIds.cend());
|
||||
}
|
||||
|
||||
vector<Shape> shapes;
|
||||
for (auto const & shape : m_shapes)
|
||||
{
|
||||
if (shapeIdInEdges.count(shape.GetId()) != 0)
|
||||
shapes.push_back(shape);
|
||||
}
|
||||
|
||||
m_shapes.swap(shapes);
|
||||
}
|
||||
} // namespace transit
|
||||
} // namespace routing
|
195
routing_common/transit_graph_data.hpp
Normal file
195
routing_common/transit_graph_data.hpp
Normal file
|
@ -0,0 +1,195 @@
|
|||
#pragma once
|
||||
|
||||
#include "routing_common/transit_types.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
#include "geometry/region2d.hpp"
|
||||
|
||||
#include "coding/reader.hpp"
|
||||
#include "coding/writer.hpp"
|
||||
|
||||
#include "base/exception.hpp"
|
||||
#include "base/osm_id.hpp"
|
||||
#include "base/visitor.hpp"
|
||||
|
||||
#include "3party/jansson/myjansson.hpp"
|
||||
|
||||
#include <cstdint>
|
||||
#include <cstring>
|
||||
#include <map>
|
||||
#include <string>
|
||||
#include <type_traits>
|
||||
#include <vector>
|
||||
|
||||
namespace routing
|
||||
{
|
||||
namespace transit
|
||||
{
|
||||
using OsmIdToFeatureIdsMap = std::map<osm::Id, std::vector<FeatureId>>;
|
||||
|
||||
class DeserializerFromJson
|
||||
{
|
||||
public:
|
||||
DeserializerFromJson(json_struct_t * node, OsmIdToFeatureIdsMap const & osmIdToFeatureIds);
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_integral<T>::value || std::is_enum<T>::value ||
|
||||
std::is_same<T, double>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
GetField(t, name);
|
||||
return;
|
||||
}
|
||||
|
||||
void operator()(std::string & s, char const * name = nullptr) { GetField(s, name); }
|
||||
void operator()(m2::PointD & p, char const * name = nullptr);
|
||||
void operator()(FeatureIdentifiers & id, char const * name = nullptr);
|
||||
void operator()(EdgeFlags & edgeFlags, char const * name = nullptr);
|
||||
void operator()(StopIdRanges & rs, char const * name = nullptr);
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_same<T, Edge::WrappedEdgeId>::value ||
|
||||
std::is_same<T, Stop::WrappedStopId>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
typename T::RepType id;
|
||||
operator()(id, name);
|
||||
t.Set(id);
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void operator()(std::vector<T> & vs, char const * name = nullptr)
|
||||
{
|
||||
auto * arr = my::GetJSONOptionalField(m_node, name);
|
||||
if (arr == nullptr)
|
||||
return;
|
||||
|
||||
if (!json_is_array(arr))
|
||||
MYTHROW(my::Json::Exception, ("The field", name, "must contain a json array."));
|
||||
size_t const sz = json_array_size(arr);
|
||||
vs.resize(sz);
|
||||
for (size_t i = 0; i < sz; ++i)
|
||||
{
|
||||
DeserializerFromJson arrayItem(json_array_get(arr, i), m_osmIdToFeatureIds);
|
||||
arrayItem(vs[i]);
|
||||
}
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
typename std::enable_if<std::is_class<T>::value &&
|
||||
!std::is_same<T, Edge::WrappedEdgeId>::value &&
|
||||
!std::is_same<T, Stop::WrappedStopId>::value>::type
|
||||
operator()(T & t, char const * name = nullptr)
|
||||
{
|
||||
if (name != nullptr && json_is_object(m_node))
|
||||
{
|
||||
json_t * dictNode = my::GetJSONOptionalField(m_node, name);
|
||||
if (dictNode == nullptr)
|
||||
return; // No such field in json.
|
||||
|
||||
DeserializerFromJson dict(dictNode, m_osmIdToFeatureIds);
|
||||
t.Visit(dict);
|
||||
return;
|
||||
}
|
||||
|
||||
t.Visit(*this);
|
||||
}
|
||||
|
||||
private:
|
||||
template <typename T>
|
||||
void GetField(T & t, char const * name = nullptr)
|
||||
{
|
||||
if (name == nullptr)
|
||||
{
|
||||
// |name| is not set in case of array items
|
||||
FromJSON(m_node, t);
|
||||
return;
|
||||
}
|
||||
|
||||
json_struct_t * field = my::GetJSONOptionalField(m_node, name);
|
||||
if (field == nullptr)
|
||||
{
|
||||
// No optional field |name| at |m_node|. In that case the default value should be set to |t|.
|
||||
// This default value is set at constructor of corresponding class which is filled with
|
||||
// |DeserializerFromJson|. And the value (|t|) is not changed at this method.
|
||||
return;
|
||||
}
|
||||
FromJSON(field, t);
|
||||
}
|
||||
|
||||
json_struct_t * m_node;
|
||||
OsmIdToFeatureIdsMap const & m_osmIdToFeatureIds;
|
||||
};
|
||||
|
||||
/// \brief The class contains all the information to make TRANSIT_FILE_TAG section.
|
||||
class GraphData
|
||||
{
|
||||
public:
|
||||
void DeserializeFromJson(my::Json const & root, OsmIdToFeatureIdsMap const & mapping);
|
||||
void Serialize(Writer & writer) const;
|
||||
void Deserialize(Reader & reader);
|
||||
void AppendTo(GraphData const & rhs);
|
||||
void Clear();
|
||||
bool IsValid() const;
|
||||
bool IsEmpty() const;
|
||||
|
||||
/// \brief Sorts all class fields by their ids.
|
||||
void Sort();
|
||||
/// \brief Removes some items from all the class fields if they are outside |borders|.
|
||||
/// Please see description for the other Clip*() method for excact rules of clipping.
|
||||
/// \note Before call of the method every line in |m_stopIds| should contain |m_stopIds|
|
||||
/// with only one stop range.
|
||||
void ClipGraph(std::vector<m2::RegionD> const & borders);
|
||||
void SetGateBestPedestrianSegment(size_t gateIdx, SingleMwmSegment const & s);
|
||||
|
||||
std::vector<Stop> const & GetStops() const { return m_stops; }
|
||||
std::vector<Gate> const & GetGates() const { return m_gates; }
|
||||
std::vector<Edge> const & GetEdges() const { return m_edges; }
|
||||
std::vector<Transfer> const & GetTransfers() const { return m_transfers; }
|
||||
std::vector<Line> const & GetLines() const { return m_lines; }
|
||||
std::vector<Shape> const & GetShapes() const { return m_shapes; }
|
||||
std::vector<Network> const & GetNetworks() const { return m_networks; }
|
||||
|
||||
private:
|
||||
DECLARE_VISITOR_AND_DEBUG_PRINT(GraphData, visitor(m_stops, "stops"), visitor(m_gates, "gates"),
|
||||
visitor(m_edges, "edges"), visitor(m_transfers, "transfers"),
|
||||
visitor(m_lines, "lines"), visitor(m_shapes, "shapes"),
|
||||
visitor(m_networks, "networks"))
|
||||
|
||||
bool IsUnique() const;
|
||||
bool IsSorted() const;
|
||||
|
||||
/// \brief Clipping |m_lines| with |borders|.
|
||||
/// \details After a call of the method the following stop ids in |m_lines| are left:
|
||||
/// * stops inside |borders|
|
||||
/// * stops which are connected with an edge from |m_edges| with stops inside |borders|
|
||||
/// \note Lines without stops are removed from |m_lines|.
|
||||
/// \note Before call of the method every line in |m_lines| should contain |m_stopIds|
|
||||
/// with only one stop range.
|
||||
void ClipLines(std::vector<m2::RegionD> const & borders);
|
||||
/// \brief Removes all stops from |m_stops| which are not contained in |m_lines| at field |m_stopIds|.
|
||||
/// \note Only stops which stop ids contained in |m_lines| will left in |m_stops|
|
||||
/// after call of this method.
|
||||
void ClipStops();
|
||||
/// \brief Removes all networks from |m_networks| which are not contained in |m_lines|
|
||||
/// at field |m_networkId|.
|
||||
void ClipNetworks();
|
||||
/// \brief Removes gates from |m_gates| if there's no stop in |m_stops| with their stop ids.
|
||||
void ClipGates();
|
||||
/// \brief Removes transfers from |m_transfers| if there's no stop in |m_stops| with their stop ids.
|
||||
void ClipTransfer();
|
||||
/// \brief Removes edges from |m_edges| if their ends are not contained in |m_stops|.
|
||||
void ClipEdges();
|
||||
/// \brief Removes all shapes from |m_shapes| which are not reffered form |m_edges|.
|
||||
void ClipShapes();
|
||||
|
||||
std::vector<Stop> m_stops;
|
||||
std::vector<Gate> m_gates;
|
||||
std::vector<Edge> m_edges;
|
||||
std::vector<Transfer> m_transfers;
|
||||
std::vector<Line> m_lines;
|
||||
std::vector<Shape> m_shapes;
|
||||
std::vector<Network> m_networks;
|
||||
};
|
||||
} // namespace transit
|
||||
} // namespace routing
|
|
@ -29,6 +29,8 @@
|
|||
34F558571DBF3CD800A4FC11 /* libstdc++.tbd in Frameworks */ = {isa = PBXBuildFile; fileRef = 34F558561DBF3CD800A4FC11 /* libstdc++.tbd */; };
|
||||
562147271F6AA36A002D2214 /* libbsdiff.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 562147261F6AA36A002D2214 /* libbsdiff.a */; };
|
||||
562147291F6AA37E002D2214 /* libmwm_diff.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 562147281F6AA37E002D2214 /* libmwm_diff.a */; };
|
||||
562D42941FD8460500A995F3 /* libugc.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 562D42951FD8460500A995F3 /* libugc.a */; };
|
||||
562D42961FD8463700A995F3 /* libsqlite3.tbd in Frameworks */ = {isa = PBXBuildFile; fileRef = 670E7BC61EF992F600A8E9ED /* libsqlite3.tbd */; };
|
||||
56E2EDAA1F7E3FBB0092E9C2 /* transit_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 56E2EDA91F7E3FBB0092E9C2 /* transit_test.cpp */; };
|
||||
670E7BBC1EF9832200A8E9ED /* libicu.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 670E7BBB1EF9832200A8E9ED /* libicu.a */; };
|
||||
670E7BBE1EF9839C00A8E9ED /* librouting_common.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 670E7BBD1EF9839C00A8E9ED /* librouting_common.a */; };
|
||||
|
@ -204,6 +206,7 @@
|
|||
34F558561DBF3CD800A4FC11 /* libstdc++.tbd */ = {isa = PBXFileReference; lastKnownFileType = "sourcecode.text-based-dylib-definition"; name = "libstdc++.tbd"; path = "usr/lib/libstdc++.tbd"; sourceTree = SDKROOT; };
|
||||
562147261F6AA36A002D2214 /* libbsdiff.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libbsdiff.a; path = ../bsdiff/build/Debug/libbsdiff.a; sourceTree = "<group>"; };
|
||||
562147281F6AA37E002D2214 /* libmwm_diff.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libmwm_diff.a; path = "../../../../Library/Developer/Xcode/DerivedData/omim-gsfdicnjgjjbizhdmwedavcucpok/Build/Products/Debug/libmwm_diff.a"; sourceTree = "<group>"; };
|
||||
562D42951FD8460500A995F3 /* libugc.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; path = libugc.a; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
56E2EDA91F7E3FBB0092E9C2 /* transit_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = transit_test.cpp; sourceTree = "<group>"; };
|
||||
670E7BBB1EF9832200A8E9ED /* libicu.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libicu.a; path = "../../../../Library/Developer/Xcode/DerivedData/omim-gzleizqujktwggdwiejzkgjrsgvp/Build/Products/Debug/libicu.a"; sourceTree = "<group>"; };
|
||||
670E7BBD1EF9839C00A8E9ED /* librouting_common.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = librouting_common.a; path = "../../../../Library/Developer/Xcode/DerivedData/omim-gzleizqujktwggdwiejzkgjrsgvp/Build/Products/Debug/librouting_common.a"; sourceTree = "<group>"; };
|
||||
|
@ -342,6 +345,8 @@
|
|||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
562D42961FD8463700A995F3 /* libsqlite3.tbd in Frameworks */,
|
||||
562D42941FD8460500A995F3 /* libugc.a in Frameworks */,
|
||||
562147291F6AA37E002D2214 /* libmwm_diff.a in Frameworks */,
|
||||
562147271F6AA36A002D2214 /* libbsdiff.a in Frameworks */,
|
||||
670E7BC01EF983A400A8E9ED /* libtraffic.a in Frameworks */,
|
||||
|
@ -430,6 +435,7 @@
|
|||
34F558551DBF3CD800A4FC11 /* Frameworks */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
562D42951FD8460500A995F3 /* libugc.a */,
|
||||
562147281F6AA37E002D2214 /* libmwm_diff.a */,
|
||||
562147261F6AA36A002D2214 /* libbsdiff.a */,
|
||||
670E7BC61EF992F600A8E9ED /* libsqlite3.tbd */,
|
||||
|
@ -773,6 +779,7 @@
|
|||
"$(inherited)",
|
||||
"$(OMIM_ROOT)/3party/gflags/src/",
|
||||
"$(OMIM_ROOT)/3party/glm",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = Debug;
|
||||
|
@ -785,6 +792,7 @@
|
|||
"$(inherited)",
|
||||
"$(OMIM_ROOT)/3party/gflags/src/",
|
||||
"$(OMIM_ROOT)/3party/glm",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = Release;
|
||||
|
@ -829,6 +837,7 @@
|
|||
"$(inherited)",
|
||||
"$(OMIM_ROOT)/3party/gflags/src/",
|
||||
"$(OMIM_ROOT)/3party/glm",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = "Production Full";
|
||||
|
|
|
@ -8,6 +8,8 @@
|
|||
|
||||
/* Begin PBXBuildFile section */
|
||||
40FF45D01F388EF80046BD40 /* vehicle_model_for_country_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 40FF45CF1F388EF80046BD40 /* vehicle_model_for_country_test.cpp */; };
|
||||
562D42921FD83FE100A995F3 /* transit_graph_data.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 562D42901FD83FE000A995F3 /* transit_graph_data.hpp */; };
|
||||
562D42931FD83FE100A995F3 /* transit_graph_data.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 562D42911FD83FE100A995F3 /* transit_graph_data.cpp */; };
|
||||
5647A4511F72BEB600DE1125 /* libicu.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5647A4521F72BEB600DE1125 /* libicu.a */; };
|
||||
5667C1DD1F751F2700C6B31B /* transit_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 5647A4531F72BF2B00DE1125 /* transit_test.cpp */; };
|
||||
56D0E47D1F8E335D0084B18C /* num_mwm_id.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 56D0E47C1F8E335D0084B18C /* num_mwm_id.hpp */; };
|
||||
|
@ -43,6 +45,8 @@
|
|||
|
||||
/* Begin PBXFileReference section */
|
||||
40FF45CF1F388EF80046BD40 /* vehicle_model_for_country_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = vehicle_model_for_country_test.cpp; sourceTree = "<group>"; };
|
||||
562D42901FD83FE000A995F3 /* transit_graph_data.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = transit_graph_data.hpp; sourceTree = "<group>"; };
|
||||
562D42911FD83FE100A995F3 /* transit_graph_data.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = transit_graph_data.cpp; sourceTree = "<group>"; };
|
||||
5647A4521F72BEB600DE1125 /* libicu.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; path = libicu.a; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
5647A4531F72BF2B00DE1125 /* transit_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = transit_test.cpp; sourceTree = "<group>"; };
|
||||
5667C1DE1F751F4200C6B31B /* routing_common.pro */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; path = routing_common.pro; sourceTree = "<group>"; };
|
||||
|
@ -138,6 +142,8 @@
|
|||
671E78741E6A3BE200B2859B /* routing_common */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
562D42911FD83FE100A995F3 /* transit_graph_data.cpp */,
|
||||
562D42901FD83FE000A995F3 /* transit_graph_data.hpp */,
|
||||
56E6E73F1F95D5690022CBD3 /* transit_speed_limits.hpp */,
|
||||
56D0E47C1F8E335D0084B18C /* num_mwm_id.hpp */,
|
||||
56E2EDA31F7E3F890092E9C2 /* transit_serdes.hpp */,
|
||||
|
@ -203,6 +209,7 @@
|
|||
56E41D881F72B42F00E28E2D /* transit_types.hpp in Headers */,
|
||||
56E2EDA61F7E3F8A0092E9C2 /* transit_serdes.hpp in Headers */,
|
||||
56E6E7401F95D5690022CBD3 /* transit_speed_limits.hpp in Headers */,
|
||||
562D42921FD83FE100A995F3 /* transit_graph_data.hpp in Headers */,
|
||||
671E78891E6A3C5D00B2859B /* bicycle_model.hpp in Headers */,
|
||||
671E788B1E6A3C5D00B2859B /* car_model.hpp in Headers */,
|
||||
671E788F1E6A3C5D00B2859B /* vehicle_model.hpp in Headers */,
|
||||
|
@ -300,6 +307,7 @@
|
|||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
56E2EDA81F7E3F8A0092E9C2 /* transit_types.cpp in Sources */,
|
||||
562D42931FD83FE100A995F3 /* transit_graph_data.cpp in Sources */,
|
||||
671E788A1E6A3C5D00B2859B /* car_model.cpp in Sources */,
|
||||
671E78881E6A3C5D00B2859B /* bicycle_model.cpp in Sources */,
|
||||
671E788E1E6A3C5D00B2859B /* vehicle_model.cpp in Sources */,
|
||||
|
@ -325,6 +333,12 @@
|
|||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 671E78901E6A3C9C00B2859B /* common-debug.xcconfig */;
|
||||
buildSettings = {
|
||||
HEADER_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"$(OMIM_ROOT)",
|
||||
"$(BOOST_ROOT)",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
|
@ -332,6 +346,12 @@
|
|||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 671E78911E6A3C9C00B2859B /* common-release.xcconfig */;
|
||||
buildSettings = {
|
||||
HEADER_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"$(OMIM_ROOT)",
|
||||
"$(BOOST_ROOT)",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
|
@ -373,6 +393,12 @@
|
|||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 671E78911E6A3C9C00B2859B /* common-release.xcconfig */;
|
||||
buildSettings = {
|
||||
HEADER_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"$(OMIM_ROOT)",
|
||||
"$(BOOST_ROOT)",
|
||||
"$(OMIM_ROOT)/3party/jansson/src",
|
||||
);
|
||||
};
|
||||
name = "Production Full";
|
||||
};
|
||||
|
|
Loading…
Add table
Reference in a new issue