Map Matching
grid.cpp
Go to the documentation of this file.
1 #include "grid.h"
2 #include <QString>
3 #include <QStringList>
4 #include <fstream>
5 #include <iomanip>
6 #include <iostream>
7 #include <regex>
8 #include <vector>
9 
10 #define DEBUG_READCSV false
11 #define DEBUG_ADDROAD false
12 
13 long Grid::counter = 0;
14 
16  : m_xMin(0.0)
17  , m_xMax(std::numeric_limits<double>::max())
18  , m_yMin(0.0)
19  , m_yMax(std::numeric_limits<double>::max())
20  , m_xMinGrid(std::numeric_limits<double>::max())
21  , m_xMaxGrid(0.0)
22  , m_yMinGrid(std::numeric_limits<double>::max())
23  , m_yMaxGrid(0.0)
24 {
25 }
26 
28 {
29  m_mapOfAllRoads.clear();
30  m_mapOfExtPoints.clear(); // should be already done at the end of reading CSV
31  m_vectorOfPoints.clear();
32 }
33 
34 void Grid::setTrackBoundingBox(double xMin, double xMax, double yMin, double yMax) // from GPS track
35 {
36  m_xMin = xMin;
37  m_xMax = xMax;
38  m_yMin = yMin;
39  m_yMax = yMax;
40 }
41 
42 void Grid::readFromCSV(QString filename)
43 {
44 
45  emit signalMessage("Loading grid ...");
46  std::string line;
47  QString stringConverted;
48  m_gridFullName = filename.toStdString();
49  // Declare file stream
50  std::ifstream file(filename.toStdString().c_str()); // c_str() http://stackoverflow.com/questions/32332/why-dont-the-stdfstream-classes-take-a-stdstring
51 
52  std::vector<int> correspondance(4, -1); // Correpondance between the header and the parser :
53  // correspondance :
54  // 0 : # WKT
55  // 1 : # Edge ID
56  // 2 : # From Node
57  // 3 : # To Node ID
58 
59  // Read header
60  if (file.good()) {
61  getline(file, line);
62  stringConverted = QString::fromStdString(line);
63  QStringList text = stringConverted.split(",");
64 
65  // Parse header
66  for (int i = 0; i < text.size(); ++i) {
67  if (text[i].contains(QString::fromStdString("WKT"),
68  Qt::CaseInsensitive)) {
69  correspondance[0] = i;
70  //cout << "WKT DETECTED at colonne : " << i << endl;
71  } else if (text[i].contains(QString::fromStdString("Edge ID"),
72  Qt::CaseInsensitive)) {
73  correspondance[1] = i;
74  //cout << "Edge ID DETECTED at colonne : " << i << endl;
75  } else if (text[i].contains(QString::fromStdString("From Node"),
76  Qt::CaseInsensitive)) {
77  correspondance[2] = i;
78  //cout << "From Node DETECTED at colonne : " << i << endl;
79  } else if (text[i].contains(QString::fromStdString("To Node ID"),
80  Qt::CaseInsensitive)) {
81  correspondance[3] = i;
82  //cout << "To Node ID DETECTED at colonne : " << i << endl;
83  } else {
84  //cout << "Colonne " << text[i].toStdString() << " non reconnue" << endl;
85  }
86  }
87  }
88 
89  // Display the correspondance table
90  /*cout << "BEGIN CORRESPONDANCE" << endl;
91  for (uint i = 0; i < correspondance.size(); ++i) {
92  cout << correspondance[i] << endl;
93  }
94  cout << "END CORRESPONDANCE" << endl;*/
95 
96  while (file.good()) {
97  long edgeId(0);
98  //long fromNodeId(0);
99  //long toNodeId(0);
100  std::vector<std::vector<double> > listOfCoordinates(0);
101 
102  getline(file, line);
103  // Convert string to Qstring (easiest
104  stringConverted = QString::fromStdString(line);
105  if (stringConverted.length() != 0) // Elimine les lignes vides
106  {
107  // Exemple :QMetaObject
108  // Entrée :
109  // "LINESTRING (1.37 3.36,1.24 3.84)",839,825,883,1,22.222222219999999,10
110  // Sortie :
111  // [ LINESTRING (1.37 3.36,1.24 3.84) , 839 , 825 , 883]
112  // A cause de la virgule dans linestring, on est obligé de :
113  // Spliter sur les virgules les éventuels éléments avant les guillemets
114  // Ajouter ce contenu à la liste
115  // Spliter sur les guillemets
116  // et récupérer la chaine entre les guillemets LINESTRING (1.37 3.36,1.24 3.84)
117  // Ajouter ce contenu à la liste
118  // Spliter sur les virgules les éventuels éléments après les guillemets
119  // Supprimer le premier élement du split (car le Linestring est déjà récupéré)
120  // Ajouter le reste à la liste
121 
122  QStringList linestringList = stringConverted.split("\"");
123 
124  //QString before = linestringList[0];
125  //QString linestring = linestringList[1];
126  //QString after = linestringList[2];
127 
128  QStringList text;
129 
130  if (!linestringList[0].isEmpty()) {
131  text = linestringList[0].split(",");
132  }
133  text.append(linestringList[1]);
134 
135  QStringList afterSplit = linestringList[2].split(",");
136  afterSplit.removeFirst();
137 
138  if (!afterSplit.isEmpty()) {
139  text += afterSplit;
140  }
141 
142  //Initialisation du test
143  bool inBox = false;
144  for (int i = 0; i < text.size(); ++i) {
145  if (i == correspondance[0]) {
146  // traitement WTK
147  QStringList contenuList = text[i].split("LINESTRING (");
148  contenuList = contenuList[1].split(")");
149  QString contenu = contenuList[0];
150 
151  QStringList listePoints = contenu.split(",");
152  if (DEBUG_READCSV)
153  std::cout << "Coordonnées points : " << std::endl;
154  for (int j = 0; j < listePoints.size(); ++j) {
155  double x(0.0);
156  double y(0.0);
157  QStringList coordonnees = listePoints[j].split(" ");
158  x = coordonnees[0].toDouble();
159  y = coordonnees[1].toDouble();
160  inBox = inFootPrint(x, y);
161  updateGrid(x, y);
162  if (DEBUG_READCSV)
163  std::cout << std::setprecision(150) << x << "," << y << std::endl;
164  std::vector<double> coordinates;
165  coordinates.push_back(x);
166  coordinates.push_back(y);
167  listOfCoordinates.push_back(coordinates);
168  }
169 
170  } else if (i == correspondance[1]) {
171  // traitement Edge ID
172  if (DEBUG_READCSV)
173  std::cout << "Edge ID : " << text[i].toStdString() << " ";
174  edgeId = text[i].toLong();
175  } else if (i == correspondance[2]) {
176  // traitement From Node
177  if (DEBUG_READCSV)
178  std::cout << "From Node : " << text[i].toStdString() << " ";
179  //fromNodeId = text[i].toLong();
180  } else if (i == correspondance[3]) {
181  // traitement To Node ID
182  if (DEBUG_READCSV)
183  std::cout << "To Node ID : " << text[i].toStdString() << " ";
184  //toNodeId = text[i].toLong();
185  }
186  }
187  if (DEBUG_READCSV)
188  std::cout << std::endl;
189  if (inBox)
190  addRoad(listOfCoordinates, edgeId);
191  }
192  }
193  // m_mapOfExtPoints.clear(); // TODO remove comment, and DO clear map (kept for test)
194 }
195 
196 void Grid::addRoad(const std::vector<std::vector<double> >& listOfCoordinates, long edgeId)
197 {
198  int n = listOfCoordinates.size();
199  int curPoint = 0; // used to apply a special treatment to first and last point of a road
200  bool newPoint = true;
201  int existingPointId = -1; // id of the point if its already exists
202  if (edgeId == 0) {
203  edgeId = counter;
204  counter++;
205  }
206  Road road(edgeId);
207  for (const auto& coord : listOfCoordinates) {
208  newPoint = true;
209  PointRoad p(coord[0], coord[1], (curPoint == 0 || curPoint == n - 1));
210  PointRoad* pp = &p;
211  if (curPoint == 0 || curPoint == n - 1) { // for extremities, check if road already exists, if so use it
212  if (m_mapOfExtPoints.count(p)) {
213  ExtremityPointMap::const_iterator got = m_mapOfExtPoints.find(p);
214  existingPointId = got->second;
215  if (DEBUG_ADDROAD)
216  p.outputInfos();
217  newPoint = false;
218  pp = &m_vectorOfPoints[existingPointId]; // TODO error check if not a new point
219  } else {
220  pp = &p;
221  }
222  }
223  pp->updateBelongToRoad(edgeId);
224  if (newPoint) {
225  p.setid(m_vectorOfPoints.size());
226  m_vectorOfPoints.push_back(std::move(p));
227  if (p.isNode())
228  m_mapOfExtPoints[p] = p.id();
229  }
230  road.addPoint(pp->id());
231  ++curPoint;
232  }
233  m_mapOfAllRoads[road.edgeId()] = road;
234  //m_mapOfAllRoads.insert({road.edgeId(), std::move(road)});
235 }
236 
238 {
239  std::cout << "Network " << m_gridFullName << " contains: \n\t" << m_mapOfAllRoads.size() << " roads" << std::endl;
240  std::cout << "\twith a grand total of " << m_vectorOfPoints.size() << " points" << std::endl;
241  std::cout << "\tof which " << m_mapOfExtPoints.size() << " are extremities." << std::endl;
242 }
243 
244 std::string Grid::infos()
245 {
246  std::stringstream ss;
247  ss << "Network " << m_gridFullName << " contains: \n\t" << m_mapOfAllRoads.size() << " roads\n";
248  ss << "\twith a grand total of " << m_vectorOfPoints.size() << " points\n";
249  ss << "\tof which " << m_mapOfExtPoints.size() << " are extremities.\n";
250  return ss.str();
251 }
252 
253 AllRoadMap::iterator Grid::getRoadEntry(long id)
254 {
255  AllRoadMap::iterator got = m_mapOfAllRoads.find(id);
256  return got;
257 }
258 
259 bool Grid::inFootPrint(double x, double y)
260 {
261  return (m_xMin <= x) && (x <= m_xMax) && (m_yMin <= y) && (y <= m_yMax);
262 }
263 
265 {
266  bool DEBUG = false;
267  // check all points and for those who are a node (extremity of a road) update all the roads
268  for (const auto& p : m_vectorOfPoints) {
269  if (p.isNode()) {
270  const std::vector<long>& listOfRoadId = p.vectorOfRoadId();
271  for (const auto roadId : listOfRoadId) {
272  AllRoadMap::iterator got = getRoadEntry(roadId);
273  for (const auto neighborId : listOfRoadId) {
274  got->second.addNeighbor(neighborId);
275  if (DEBUG) {
276  std::cout << "update neighbour " << std::endl;
277  got->second.outputInfos();
278  }
279  }
280  }
281  }
282  }
283  // verification
284 
285  if (DEBUG) {
286  std::cout << "Resutat: " << std::endl;
287  for (auto& it : m_mapOfAllRoads) {
288  it.second.outputInfos();
289  }
290  }
291 }
292 
293 double Grid::computeDistanceFraction(PointGPS* prevPoint, PointGPS* curPoint, long prevRoadId, long curRoadId)
294 {
295  double dbird, droad; // distance
296  double r = 0.0; // fraction
297  AllRoadMap::iterator got = getRoadEntry(prevRoadId);
298  Road* r1 = &got->second;
299  Road* r2 = &getRoadEntry(prevRoadId)->second;
300  // find projection of each point on road
301  std::vector<double> k1 = getProjectedPointAndDistance(prevPoint, r1);
302  std::vector<double> k2 = getProjectedPointAndDistance(curPoint, r2);
303  Point projR1(k1.at(0), k1.at(1)), projR2(k2.at(0), k2.at(1));
304  int node = r1->getIntersectionIDWith(r2);
305  // bird distance
306  dbird = prevPoint->distanceToPoint(*curPoint);
307  if (prevRoadId == curRoadId) {
308  droad = getDistanceBetweenProjections(&projR1, &projR2, r1);
309  } else {
310  droad = getDistanceToExtremity(&projR1, node, r1) + getDistanceToExtremity(&projR2, node, r2);
311  }
312 
313  //Point *p= r1->
314  return r;
315 }
316 
318 {
319  std::vector<double> res, bestRes;
320  double d, bestDistance = std::numeric_limits<double>::max();
321  const std::vector<int>& listOfPointId = r->vectorOfPointsId();
322  for (uint i = 1; i < listOfPointId.size(); i++) {
323  res = p->projectionOnSegment(m_vectorOfPoints.at(listOfPointId[i - 1]), m_vectorOfPoints.at(listOfPointId[i]));
324  d = res.at(2);
325  if (d < bestDistance) {
326  bestDistance = d;
327  bestRes = res;
328  }
329  }
330  return res;
331 }
332 
334 {
335  double d1, d2, dTotal = 0.0;
336  const std::vector<int>& listOfPointId = r1->vectorOfPointsId();
337  int start1 = getSegmentCounter(projR1, r1);
338  int start2 = getSegmentCounter(projR2, r1);
339 
340  if (start1 == start2)
341  return projR1->distanceToPoint(*projR2);
342  // compute segments inbetween
343  if (start1 < start2) {
344  // from p1 to end of segment, all segment inbetween then from start2-1 to p2
345  } else {
346  }
347  // TODO
348  return dTotal;
349 }
350 
351 double Grid::getDistanceToExtremity(Point* projR1, int node, Road* r1)
352 {
353  double d, dTotal = 0.0;
354  const std::vector<int>& listOfPointId = r1->vectorOfPointsId();
355  // start with extremity and stop when d==0
356  if (listOfPointId.at(0) == node) {
357  for (uint i = 1; i < listOfPointId.size(); i++) {
358  d = projR1->distanceToSegment(m_vectorOfPoints.at(listOfPointId[i - 1]), m_vectorOfPoints.at(listOfPointId[i]));
359  dTotal += d;
360  if (d < 0.0001)
361  break;
362  }
363  } else {
364  for (uint i = listOfPointId.size() - 1; i > 1; i--) {
365  d = projR1->distanceToSegment(m_vectorOfPoints.at(listOfPointId[i - 1]), m_vectorOfPoints.at(listOfPointId[i]));
366  dTotal += d;
367  if (d < 0.0001)
368  break;
369  }
370  }
371  return dTotal;
372 }
373 
375 {
376  double d;
377  const std::vector<int>& listOfPointId = r->vectorOfPointsId();
378  for (uint i = 1; i < listOfPointId.size(); i++) {
379  d = p->distanceToSegment(m_vectorOfPoints.at(listOfPointId[i - 1]), m_vectorOfPoints.at(listOfPointId[i]));
380  if (d <= 0.0001)
381  return i;
382  }
383 }
384 
385 void Grid::updateGrid(double x, double y)
386 {
387  m_xMinGrid = std::min(m_xMinGrid, x);
388  m_xMaxGrid = std::max(m_xMaxGrid, x);
389  m_yMinGrid = std::min(m_yMinGrid, y);
390  m_yMaxGrid = std::max(m_yMaxGrid, y);
391 }
392 
394 {
395  return (xMin() >= xMinGrid()) && (xMax() <= xMaxGrid()) && (yMin() >= yMinGrid()) && (yMax() <= yMaxGrid());
396 }
void buildMarkovMatrix()
buildMarkovMatrix
Definition: grid.cpp:264
double distanceToSegment(const Point &p1, const Point &p2) const
Compute the distance between a point and a segment.
Definition: point.cpp:32
std::string m_gridFullName
Definition: grid.h:136
double getDistanceToExtremity(Point *projR1, int node, Road *r1)
Definition: grid.cpp:351
bool trackInGrid()
Check if track is in grid.
Definition: grid.cpp:393
virtual ~Grid()
Destructor.
Definition: grid.cpp:27
void outputInfos()
Definition: pointroad.cpp:12
double xMax() const
Definition: grid.h:121
double xMin() const
Definition: grid.h:120
static long counter
Definition: grid.h:138
void setid(int id)
Definition: pointroad.h:34
#define DEBUG_READCSV
Definition: grid.cpp:10
const std::vector< int > & vectorOfPointsId() const
Get the vector of points composing the road.
Definition: road.h:71
double yMaxGrid() const
Definition: grid.h:127
double m_xMinGrid
Definition: grid.h:154
void addRoad(const std::vector< std::vector< double > > &listOfCoordinates, long edgeId)
Creates a new road and inserts it in m_road.
Definition: grid.cpp:196
STL namespace.
double m_xMin
The coordinates of the track.
Definition: grid.h:144
void updateBelongToRoad(long roadId)
Definition: pointroad.cpp:7
double yMax() const
Definition: grid.h:123
AllRoadMap m_mapOfAllRoads
Definition: grid.h:129
Road is an element of a network. Road are strongly linked with Points.
Definition: road.h:14
double yMin() const
Definition: grid.h:122
double xMaxGrid() const
Definition: grid.h:125
bool inFootPrint(double x, double y)
Test if a point is in the defined area of a track.
Definition: grid.cpp:259
void updateGrid(double x, double y)
Find max and min of the grid.
Definition: grid.cpp:385
double m_yMin
Definition: grid.h:146
std::vector< double > getProjectedPointAndDistance(PointGPS *p, Road *r)
Definition: grid.cpp:317
Grid()
Constructor.
Definition: grid.cpp:15
void setTrackBoundingBox(double xMin, double xMax, double yMin, double yMax)
Save data about our area of roads.
Definition: grid.cpp:34
std::vector< PointRoad > m_vectorOfPoints
Definition: grid.h:130
double distanceToPoint(const Point &p) const
Calculate distance between two points.
Definition: point.cpp:14
void outputInfos()
Definition: grid.cpp:237
double getDistanceBetweenProjections(Point *projR1, Point *projR2, Road *r1)
Definition: grid.cpp:333
int getIntersectionIDWith(Road *r) const
Definition: road.cpp:19
void addPoint(int pointId)
Add a point to the road.
Definition: road.cpp:9
double m_xMax
Definition: grid.h:145
std::string infos()
Definition: grid.cpp:244
AllRoadMap::iterator getRoadEntry(long id)
Definition: grid.cpp:253
double m_yMaxGrid
Definition: grid.h:157
double yMinGrid() const
Definition: grid.h:126
long isNode() const
Definition: pointroad.h:35
void signalMessage(QString)
ExtremityPointMap m_mapOfExtPoints
Definition: grid.h:137
long edgeId() const
Get the id of the road.
Definition: road.h:65
double m_xMaxGrid
Definition: grid.h:155
double computeDistanceFraction(PointGPS *prevPoint, PointGPS *curPoint, long prevRoadId, long curRoadId)
Definition: grid.cpp:293
double m_yMax
Definition: grid.h:147
std::vector< double > projectionOnSegment(const Point &a, const Point &b) const
Definition: point.cpp:19
int getSegmentCounter(Point *p, Road *r)
Definition: grid.cpp:374
#define DEBUG_ADDROAD
Definition: grid.cpp:11
void readFromCSV(QString filename)
Reads a csv file and inserts info into the corresponding attributs.
Definition: grid.cpp:42
The Point class.
Definition: point.h:20
int id() const
Definition: pointroad.h:33
double xMinGrid() const
Definition: grid.h:124
double m_yMinGrid
Definition: grid.h:156